顺序GUID是唯一的,但是是通过订单创建的;该顺序略有异常,与使用标准.NET Guid比较器时的顺序不同。
我正在寻找一个按顺序GUID规则排序的C#Guid比较器。
==更新==
我特别指的是由SQL Server中的NewSequentialId()创建的顺序GUID,虽然我现在意识到标准的Win32 API调用UuidCreateSequential()使用不同的SQL Server方案(我假设它们是相同的我写了这个问题。)
==更新2 ==
petelids给出了以下答案,例如, List< System.Data.SqlGuid> .Sort()给出以下序列(使用每个4位位置1的GUID的初始列表)...
01000000-0000-0000-0000-000000000000
10000000-0000-0000-0000-000000000000
00010000-0000-0000-0000-000000000000
00100000-0000-0000-0000-000000000000
00000100-0000-0000-0000-000000000000
00001000-0000-0000-0000-000000000000
00000001-0000-0000-0000-000000000000
00000010-0000-0000-0000-000000000000
00000000-0100-0000-0000-000000000000
00000000-1000-0000-0000-000000000000
00000000-0001-0000-0000-000000000000
00000000-0010-0000-0000-000000000000
00000000-0000-0100-0000-000000000000
00000000-0000-1000-0000-000000000000
00000000-0000-0001-0000-000000000000
00000000-0000-0010-0000-000000000000
00000000-0000-0000-0001-000000000000
00000000-0000-0000-0010-000000000000
00000000-0000-0000-0100-000000000000
00000000-0000-0000-1000-000000000000
00000000-0000-0000-0000-000000000001
00000000-0000-0000-0000-000000000010
00000000-0000-0000-0000-000000000100
00000000-0000-0000-0000-000000001000
00000000-0000-0000-0000-000000010000
00000000-0000-0000-0000-000000100000
00000000-0000-0000-0000-000001000000
00000000-0000-0000-0000-000010000000
00000000-0000-0000-0000-000100000000
00000000-0000-0000-0000-001000000000
00000000-0000-0000-0000-010000000000
00000000-0000-0000-0000-100000000000
与List< System.Guid> .Sort()
返回的以下顺序相反00000000-0000-0000-0000-000000000001
00000000-0000-0000-0000-000000000010
00000000-0000-0000-0000-000000000100
00000000-0000-0000-0000-000000001000
00000000-0000-0000-0000-000000010000
00000000-0000-0000-0000-000000100000
00000000-0000-0000-0000-000001000000
00000000-0000-0000-0000-000010000000
00000000-0000-0000-0000-000100000000
00000000-0000-0000-0000-001000000000
00000000-0000-0000-0000-010000000000
00000000-0000-0000-0000-100000000000
00000000-0000-0000-0001-000000000000
00000000-0000-0000-0010-000000000000
00000000-0000-0000-0100-000000000000
00000000-0000-0000-1000-000000000000
00000000-0000-0001-0000-000000000000
00000000-0000-0010-0000-000000000000
00000000-0000-0100-0000-000000000000
00000000-0000-1000-0000-000000000000
00000000-0001-0000-0000-000000000000
00000000-0010-0000-0000-000000000000
00000000-0100-0000-0000-000000000000
00000000-1000-0000-0000-000000000000
00000001-0000-0000-0000-000000000000
00000010-0000-0000-0000-000000000000
00000100-0000-0000-0000-000000000000
00001000-0000-0000-0000-000000000000
00010000-0000-0000-0000-000000000000
00100000-0000-0000-0000-000000000000
01000000-0000-0000-0000-000000000000
10000000-0000-0000-0000-000000000000
答案 0 :(得分:8)
Sql server和.NET sort guids的方式有所不同。
.NET框架中有一个名为SqlGuid
的结构,其行为应与Sql Server中的guids相同。
考虑以下从here改编的示例:
List<Guid> a = new List<Guid>();
a.Add(new Guid("3AAAAAAA-BBBB-CCCC-DDDD-2EEEEEEEEEEE"));
a.Add(new Guid("2AAAAAAA-BBBB-CCCC-DDDD-1EEEEEEEEEEE"));
a.Add(new Guid("1AAAAAAA-BBBB-CCCC-DDDD-3EEEEEEEEEEE"));
Console.WriteLine("--Unsorted Guids--");
foreach (Guid g in a)
{
Console.WriteLine("{0}", g);
}
a.Sort();
Console.WriteLine("--Sorted Guids--");
foreach (Guid g in a)
{
Console.WriteLine("{0}", g);
}
List<SqlGuid> b = new List<SqlGuid>();
b.Add(new SqlGuid("3AAAAAAA-BBBB-CCCC-DDDD-2EEEEEEEEEEE"));
b.Add(new SqlGuid("2AAAAAAA-BBBB-CCCC-DDDD-1EEEEEEEEEEE"));
b.Add(new SqlGuid("1AAAAAAA-BBBB-CCCC-DDDD-3EEEEEEEEEEE"));
b.Sort();
Console.WriteLine("--Sorted SqlGuids--");
foreach (SqlGuid sg in b)
{
Console.WriteLine("{0}", sg);
}
这会产生输出:
- 未分类的Guids--
3aaaaaaa-BBBB - CCCC-DDDD-2eeeeeeeeeee
2aaaaaaa-BBBB - CCCC-DDDD-1eeeeeeeeeee
1aaaaaaa-BBBB - CCCC-DDDD-3eeeeeeeeeee
- 各种各样的指导 -
1aaaaaaa-BBBB - CCCC-DDDD-3eeeeeeeeeee
2aaaaaaa-BBBB - CCCC-DDDD-1eeeeeeeeeee
3aaaaaaa-BBBB - CCCC-DDDD-2eeeeeeeeeee
- 排序的SqlGuids--
2aaaaaaa-BBBB - CCCC-DDDD-1eeeeeeeeeee
3aaaaaaa-BBBB - CCCC-DDDD-2eeeeeeeeeee
1aaaaaaa-BBBB-CCCC-DDDD-3eeeeeeeeeee
SqlGuid
类有一个构造函数,它接受Guid
并且从一个到另一个的转换也有效,所以它们之间的转换应该很容易。将以下代码添加到上面的代码中:
List<SqlGuid> c = a.Select(g => new SqlGuid(g)).ToList();
c.Sort();
Console.WriteLine("--Sorted SqlGuids 2--");
foreach (SqlGuid sg2 in c)
{
Console.WriteLine("{0}", sg2);
}
添加输出:
- 排序的SqlGuids 2--
2aaaaaaa-BBBB - CCCC-DDDD-1eeeeeeeeeee
3aaaaaaa-BBBB - CCCC-DDDD-2eeeeeeeeeee
1aaaaaaa-BBBB-CCCC-DDDD-3eeeeeeeeeee
答案 1 :(得分:1)
Necromancing:
答案包括如何,但不是原因
因此,只是为了记录,SQL服务器按顺序按字节对它们进行排序,也就是说自定义字节顺序:
private static readonly int[] x_rgiGuidOrder = new int[16] // 16 Bytes = 128 Bit
{10, 11, 12, 13, 14, 15, 8, 9, 6, 7, 4, 5, 0, 1, 2, 3};
换句话说,如果您将Guid想象为连续的UInt128-number,则需要将其分为16个256的基本块,并按排序顺序排列这些块以生成与SQL兼容的UID。
如果不清楚:
public class SqlGuid
: System.IComparable
, System.IComparable<SqlGuid>
, System.Collections.Generic.IComparer<SqlGuid>
, System.IEquatable<SqlGuid>
{
private const int NUM_BYTES_IN_GUID = 16;
// Comparison orders.
private static readonly int[] m_byteOrder = new int[16] // 16 Bytes = 128 Bit
{10, 11, 12, 13, 14, 15, 8, 9, 6, 7, 4, 5, 0, 1, 2, 3};
private byte[] m_bytes; // the SqlGuid is null if m_value is null
public SqlGuid(byte[] guidBytes)
{
if (guidBytes == null || guidBytes.Length != NUM_BYTES_IN_GUID)
throw new System.ArgumentException("Invalid array size");
m_bytes = new byte[NUM_BYTES_IN_GUID];
guidBytes.CopyTo(m_bytes, 0);
}
public SqlGuid(System.Guid g)
{
m_bytes = g.ToByteArray();
}
public byte[] ToByteArray()
{
byte[] ret = new byte[NUM_BYTES_IN_GUID];
m_bytes.CopyTo(ret, 0);
return ret;
}
int CompareTo(object obj)
{
if (obj == null)
return 1; // https://msdn.microsoft.com/en-us/library/system.icomparable.compareto(v=vs.110).aspx
System.Type t = obj.GetType();
if (object.ReferenceEquals(t, typeof(System.DBNull)))
return 1;
if (object.ReferenceEquals(t, typeof(SqlGuid)))
{
SqlGuid ui = (SqlGuid)obj;
return this.Compare(this, ui);
} // End if (object.ReferenceEquals(t, typeof(UInt128)))
return 1;
} // End Function CompareTo(object obj)
int System.IComparable.CompareTo(object obj)
{
return this.CompareTo(obj);
}
int CompareTo(SqlGuid other)
{
return this.Compare(this, other);
}
int System.IComparable<SqlGuid>.CompareTo(SqlGuid other)
{
return this.Compare(this, other);
}
enum EComparison : int
{
LT = -1, // itemA precedes itemB in the sort order.
EQ = 0, // itemA occurs in the same position as itemB in the sort order.
GT = 1 // itemA follows itemB in the sort order.
}
public int Compare(SqlGuid x, SqlGuid y)
{
byte byte1, byte2;
//Swap to the correct order to be compared
for (int i = 0; i < NUM_BYTES_IN_GUID; i++)
{
byte1 = x.m_bytes[m_byteOrder[i]];
byte2 = y.m_bytes[m_byteOrder[i]];
if (byte1 != byte2)
return (byte1 < byte2) ? (int) EComparison.LT : (int) EComparison.GT;
} // Next i
return (int) EComparison.EQ;
}
int System.Collections.Generic.IComparer<SqlGuid>.Compare(SqlGuid x, SqlGuid y)
{
return this.Compare(x, y);
}
public bool Equals(SqlGuid other)
{
return Compare(this, other) == 0;
}
bool System.IEquatable<SqlGuid>.Equals(SqlGuid other)
{
return this.Equals(other);
}
}
这意味着您可以在没有SqlGuid的情况下执行此操作:
public class TestClass
{
public static void Test()
{
System.Collections.Generic.List<System.Guid> ls = new System.Collections.Generic.List<System.Guid>();
for(int i = 0; i < 100; ++i)
ls.Add(System.Guid.NewGuid());
ls.Sort(Compare);
}
public static int Compare(System.Guid x, System.Guid y)
{
const int NUM_BYTES_IN_GUID = 16;
byte byte1, byte2;
byte[] xBytes = new byte[NUM_BYTES_IN_GUID];
byte[] yBytes = new byte[NUM_BYTES_IN_GUID];
x.ToByteArray().CopyTo(xBytes, 0);
y.ToByteArray().CopyTo(yBytes, 0);
int[] byteOrder = new int[16] // 16 Bytes = 128 Bit
{10, 11, 12, 13, 14, 15, 8, 9, 6, 7, 4, 5, 0, 1, 2, 3};
//Swap to the correct order to be compared
for (int i = 0; i < NUM_BYTES_IN_GUID; i++)
{
byte1 = xBytes[byteOrder[i]];
byte2 = yBytes[byteOrder[i]];
if (byte1 != byte2)
return (byte1 < byte2) ? -1 : 1;
} // Next i
return 0;
}
}
虽然使用SqlGuid会更有效,但因为SqlGuid不需要每次重新计算字节数组进行比较。