How is a guid actually stored and sorted/compared in SQL Server?

前端 未结 2 403
长情又很酷
长情又很酷 2021-01-13 10:13

Say I have this guid:

{2A87E3E2-2B6A-4149-9F5A-1B76092843D9}

Does it actually store this an an alpha numeric in the database? (I don\'t think so cause it ha

相关标签:
2条回答
  • 2021-01-13 10:49

    A GUID is stored as binary(16) internally. "Using uniqueidentifier Data" on MSDN tells you this. The { } and - are not part of the value.

    GUIDs can be "sorted" and have greater/lesser comparisons: see the canonical "How are GUIDs sorted by SQL Server?".

    Note: this means they don't sort as binary(16) (unless you CAST I suppose...)

    I can't see why you'd want this in real life (not indexing, I mean real world): about the only use for the "sorting" potential of NEWID is for the random rows tricks of ORDER BY NEWID()

    You can have "ascending" GUIDs (based on the article above) with NEWSEQUENTIALID. However, this "sorting" doesn't continue after a Windows restart = pointless IMO.

    0 讨论(0)
  • 2021-01-13 10:51

    Necromancing.
    The GUID is stored as byte-array, that is to say binary(16).
    You could also store it as UInt128, or two bigints.

    As to how a GUID is sorted:
    The code speaks for itselfs, the magical parts are

    System.Guid g
    g.ToByteArray();
    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};
    
    
    public int Compare(Guid x, Guid y)
    {
        byte byte1, byte2;
    
        //Swap to the correct order to be compared
        for (int i = 0; i < NUM_BYTES_IN_GUID; i++)
        {
            byte1 = x.ToByteArray()[m_byteOrder[i]];
            byte2 = y.ToByteArray()[m_byteOrder[i]];
            if (byte1 != byte2)
                return (byte1 < byte2) ? (int)EComparison.LT : (int)EComparison.GT;
        } // Next i 
    
        return (int)EComparison.EQ;
    }
    

    Full code:

    namespace BlueMine.Data
    {
    
    
        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);
            }
    
    
        }
    
    
    }
    
    0 讨论(0)
提交回复
热议问题