by Stan Drapkin
-
Guid FastGuid.NewGuid()
- Returns a cryptographically random GUID.
- ~10x (900%) faster than Guid.NewGuid().
-
FastGuid.Fill(Span<byte> data)
- Fills a span with cryptographically strong random bytes.
- ~1.5x (50%) to 9x (800%) faster for <512 bytes, otherwise calls RandomNumberGenerator.Fill()
-
Guid FastGuid.NewSqlServerGuid()
- Returns new Guid optimized for use as a SQL-Server clustered key.
- Guid structure is
[8 random bytes]
[8 bytes of SQL-Server-ordered DateTime.UtcNow]
- Each Guid is sequential accross 100-nanosecond
UtcNow
precision limits. - 64-bit cryptographic randomness adds uniqueness for timestamp collisions and provides reasonable unguessability and protection against online brute-force attacks.
-
Guid FastGuid.NewPostgreSqlGuid()
- Returns new Guid optimized for use as a PostgreSQL primary key or index.
- Guid structure is
[8 bytes of PostgreSQL-ordered DateTime.UtcNow]
[8 random bytes]
- Each Guid is sequential accross 100-nanosecond
UtcNow
precision limits. - 64-bit cryptographic randomness adds uniqueness for timestamp collisions and provides reasonable unguessability and protection against online brute-force attacks.
-
Helper methods for Guids generated by
NewSqlServerGuid()
:DateTime FastGuid.SqlServer.GetTimestamp(Guid guid)
- Extracts SqlServer Guid creation timestamp (UTC). Full
DateTime.UtcNow
precision.
- Extracts SqlServer Guid creation timestamp (UTC). Full
Guid FastGuid.SqlServer.MinGuidForTimestamp(DateTime timestampUtc)
Guid FastGuid.SqlServer.MaxGuidForTimestamp(DateTime timestampUtc)
- Return the smallest/largest Guid for a given timestamp (useful for time-based SQL-Server range searches).
-
Helper methods for Guids generated by
NewPostgreSqlGuid()
:DateTime FastGuid.PostgreSql.GetTimestamp(Guid guid)
- Extracts PostgreSQL Guid creation timestamp (UTC). Full
DateTime.UtcNow
precision.
- Extracts PostgreSQL Guid creation timestamp (UTC). Full
Guid FastGuid.PostgreSql.MinGuidForTimestamp(DateTime timestampUtc)
Guid FastGuid.PostgreSql.MaxGuidForTimestamp(DateTime timestampUtc)
- Return the smallest/largest Guid for a given timestamp (useful for time-based PostgreSQL range searches).
Replace all calls to Guid.NewGuid()
with FastGuid.NewGuid()
..from this:
Guid guid = Guid.NewGuid(); // your current code
..to this:
// using SecurityDriven;
Guid guid = FastGuid.NewGuid(); // 10x faster
- Thread-safe
- 128 bits of cryptographically-strong randomness
Switch from this:
Span<byte> key = stackalloc byte[32];
RandomNumberGenerator.Fill(key); // 145 nanoseconds
..to this:
Span<byte> key = stackalloc byte[32];
FastGuid.Fill(key); // 20 nanoseconds (7x faster)
Important
Guid.CreateVersion7()
causes page-fragmentation in SQL-Server and PostgreSQL.
DO NOT USE Guid.CreateVersion7()
for database Guids. Yes, I have tested it with
SQL-Server and PostgreSQL and it causes severe page-fragmentation.
Use FastGuid.NewSqlServerGuid()
or FastGuid.NewPostgreSqlGuid()
instead.
Internet advice to use Guid.CreateVersion7()
for database Guids is wrong.
public class Bench
{
[Benchmark(Baseline = true)]
public void FastGuid_NewGuid() // 12 calls
{
FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid();
FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid();
FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid(); FastGuid.NewGuid();
}
[Benchmark]
public void Guid_NewGuid() // 12 calls
{
Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid();
Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid();
Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid(); Guid.NewGuid();
}
}//class Bench
BenchmarkDotNet=v0.13.2, OS=Windows 10 (10.0.19045.2364)
Intel Core i7-10510U CPU 1.80GHz, 1 CPU, 8 logical and 4 physical cores
.NET SDK=7.0.101
[Host] : .NET 7.0.1 (7.0.122.56804), X64 RyuJIT AVX2
Method | Mean | Error | StdDev | Ratio |
---|---|---|---|---|
FastGuid_NewGuid | 116.6 ns | 2.26 ns | 5.19 ns | 1.00 |
Guid_NewGuid | 1,215.9 ns | 23.85 ns | 45.96 ns | 10.39 |
static Guid Guid_NewGuid() => Guid.NewGuid();
static Guid Guid_CreateVersion7() => Guid.CreateVersion7();
static Guid FastGuid_NewGuid() => SecurityDriven.FastGuid.NewGuid();
static Guid FastGuid_NewSqlServerGuid() => SecurityDriven.FastGuid.NewSqlServerGuid();
static Guid FastGuid_NewPostgreSqlGuid() => SecurityDriven.FastGuid.NewPostgreSqlGuid();
BenchmarkDotNet v0.13.8, Windows 10 (10.0.19045.5371/22H2/2022Update)
Intel Core i7-10510U CPU 1.80GHz, 1 CPU, 8 logical and 4 physical cores
.NET SDK 9.0.102
[Host] : .NET 9.0.1 (9.0.124.61010), X64 RyuJIT AVX2
Method | Mean | Error | StdDev | Ratio |
---|---|---|---|---|
Guid_NewGuid | 84.479 ns | 1.5090 ns | 3.3751 ns | 9x |
Guid_CreateVersion7 | 116.798 ns | 1.5795 ns | 1.8190 ns | 12x |
FastGuid_NewGuid | 9.676 ns | 0.2318 ns | 0.5278 ns | 1x |
FastGuid_NewSqlServerGuid | 36.710 ns | 0.6097 ns | 0.8546 ns | 4x |
FastGuid_NewPostgreSqlGuid | 37.292 ns | 0.7558 ns | 1.0839 ns | 4x |