0

我有以下情况:我正在创建一个可以在本地工作的数据库,但可以迁移到具有相同架构并且已经有一些数据的在线数据库。出于这个原因,我计划使用 GUID 作为它的表的主键,因为它可以通过避免更改主键值(并将其级联到引用表)的需要来简化迁移,如果我使用任何身份会发生这种情况列(我说得对吗?)。该数据库基于 SQL Server 2012 构建。

尽管如此,我希望有一个聚集索引来加快对数据库表的查询,并且当我阅读了许多文章和 Stack Overflow 答案时,我确信在 GUID 主键上使用聚集索引不是一个好主意. 所以,问题是:有没有什么好的解决方案可以在这个数据库表上建立一个聚集索引?我愿意删除 GUID 并添加另一种数据类型。一瞬间我想到了使用 HiLo 方法,但我认为这会使迁移变得更加困难,这是一个选择,但我需要一个非常好的理由来选择它(即,如果没有使用 GUID 的好方法广告 PK 并加快查询速度)。

直到现在我想到了这些解决方案:

  • 使用 newsequentialid() 生成新的 GUID,使它们成为连续的并且更好地作为聚集索引。这里的问题是:这真的比使用 int 主键更好吗?这里使用 GUID 的主要原因是帮助最终发生的任何迁移,我认为让它们按顺序排列在这里无济于事。另外,我必须检索数据库生成的 GUID,而不是在客户端创建它。
  • 使用COMB GUID并使其也成为聚集索引。似乎解决了前一种方法存在的一些问题,同时保持了较高的“随机性因子”,但不知道有没有结果。在那儿?
  • 添加一个标识 int 列并将其用作聚集索引。我的问题是它是否对任何事情都有帮助,因为无论如何查询都将使用 PK 值进行,这是一个 GUID。
  • 稍有机会使用以前的方法,将标识 int 转换为主键值和聚集索引,并添加一个 GUID 列,我将使用它进行查询。我在 Adventure Works 2012 上看到这是选择,但我真的不明白它在哪里有帮助......如果我使用 GUID(不是 PK 也不是聚集的)查询值,聚集索引会有所帮助吗?

我想这就是我能想到的全部,我真的倾向于选择 COMB 方法,并考虑一些实验来验证它是否更好以及为什么更好。但是在这种情况下有更好的解决方案吗?

4

1 回答 1

1

// 另外,我必须检索数据库生成的 GUID,而不是在客户端创建它。//

(((请注意,我想将此作为评论,但“代码”对于简单评论来说太大了。但这〜可能是一个答案。)))

您可以创建客户端(主要是)顺序 guid ..... 作为 C# 开发人员而不是 DBA,我喜欢在数据库之外创建所有关系,然后将所有内容作为 Xml 发送,让 TSQL 分解它,然后在一个 db hit 中插入我需要的所有内容(我猜偶尔也会删除)。但这种方法也有不足之处。我认为 int 和 bigint 通常会“赢得”表演比赛......但是........大多数时候............在客户端“顺序”使用 guid足以满足我的需求。特别是,正如你所建议的......你必须在某个时候移动这些数据。使用 IDENTITY() 让 PK/FK 的“阵容”是可行的,但使用 GUID(恕我直言)更容易。现在,如果我正在编写 T~ick3t M@ster 或 Am@zing 在线零售网站,我不会选择 GUID。

祝你好运。

using System;
using System.Runtime.InteropServices;


namespace MyCompany.MyTechnology.Framework.CrossDomain.GuidExtend
{
    public static class Guid
    {

        /*

        Original Reference for Code:
        http://www.pinvoke.net/default.aspx/rpcrt4/UuidCreateSequential.html

        */


        [DllImport("rpcrt4.dll", SetLastError = true)]
        static extern int UuidCreateSequential(out System.Guid guid);

        public static System.Guid NewGuid()
        {
            return CreateSequentialUuid();
        }


        public static System.Guid CreateSequentialUuid()
        {
            const int RPC_S_OK = 0;
            System.Guid g;
            int hr = UuidCreateSequential(out g);
            if (hr != RPC_S_OK)
                throw new ApplicationException("UuidCreateSequential failed: " + hr);
            return g;
        }


        /*

        Text From URL above:

        UuidCreateSequential (rpcrt4)

        Type a page name and press Enter. You'll jump to the page if it exists, or you can create it if it doesn't.
        To create a page in a module other than rpcrt4, prefix the name with the module name and a period.
        . Summary
        Creates a new UUID 
        C# Signature:
        [DllImport("rpcrt4.dll", SetLastError=true)]
        static extern int UuidCreateSequential(out Guid guid);


        VB Signature:
        Declare Function UuidCreateSequential Lib "rpcrt4.dll" (ByRef id As Guid) As Integer


        User-Defined Types:
        None.

        Notes:
        Microsoft changed the UuidCreate function so it no longer uses the machine's MAC address as part of the UUID. Since CoCreateGuid calls UuidCreate to get its GUID, its output also changed. If you still like the GUIDs to be generated in sequential order (helpful for keeping a related group of GUIDs together in the system registry), you can use the UuidCreateSequential function.

        CoCreateGuid generates random-looking GUIDs like these:

        92E60A8A-2A99-4F53-9A71-AC69BD7E4D75
        BB88FD63-DAC2-4B15-8ADF-1D502E64B92F
        28F8800C-C804-4F0F-B6F1-24BFC4D4EE80
        EBD133A6-6CF3-4ADA-B723-A8177B70D268
        B10A35C0-F012-4EC1-9D24-3CC91D2B7122



        UuidCreateSequential generates sequential GUIDs like these:

        19F287B4-8830-11D9-8BFC-000CF1ADC5B7
        19F287B5-8830-11D9-8BFC-000CF1ADC5B7
        19F287B6-8830-11D9-8BFC-000CF1ADC5B7
        19F287B7-8830-11D9-8BFC-000CF1ADC5B7
        19F287B8-8830-11D9-8BFC-000CF1ADC5B7



        Here is a summary of the differences in the output of UuidCreateSequential:

        The last six bytes reveal your MAC address 
        Several GUIDs generated in a row are sequential 
        Tips & Tricks:
        Please add some!

        Sample Code in C#:
        static Guid UuidCreateSequential()
        {
           const int RPC_S_OK = 0;
           Guid g;
           int hr = UuidCreateSequential(out g);
           if (hr != RPC_S_OK)
             throw new ApplicationException
               ("UuidCreateSequential failed: " + hr);
           return g;
        }



        Sample Code in VB:
        Sub Main()
           Dim myId As Guid
           Dim code As Integer
           code = UuidCreateSequential(myId)
           If code <> 0 Then
             Console.WriteLine("UuidCreateSequential failed: {0}", code)
           Else
             Console.WriteLine(myId)
           End If
        End Sub




        */








    }
}
于 2013-08-23T14:25:48.050 回答