我有以下情况:我正在创建一个可以在本地工作的数据库,但可以迁移到具有相同模式并且已经有一些数据的在线数据库。出于这个原因,我计划使用GUID作为它的表的主键,因为这将通过避免更改主键值(并将其级联到引用表)来简化迁移,如果我使用任何标识列(我是对的吗?)。该数据库基于SQL Server 2012构建。
尽管如此,我还是希望有一个聚集索引来加快对数据库表的查询,当我阅读了许多文章和Stack Overflow的答案时,我确信在GUID主键上有聚集索引不是一个好主意。所以,问题是:有什么好的解决方案可以在这个数据库表上建立聚集索引吗?我愿意删除GUID并添加另一种数据类型。有一段时间,我想过使用HiLo方法,但我认为这会使迁移更加困难,这是一种选择,但我需要一个非常好的理由来选择它(例如,如果没有好的方法来使用GUID ad PK并加快查询速度)。
直到现在,我想到了这些解决方案:
- 使用newsequentialid()生成新的GUID,使它们按顺序排列,并更好地作为聚集索引。这里的问题是:这真的比使用int主键好吗?这里使用GUID的主要原因是为了帮助最终发生的任何迁移,我认为使它们具有序列性在这里没有帮助。另外,我必须检索数据库生成的GUID,而不是在客户端创建它
- 使用COMB GUID并将其作为聚集索引。它似乎解决了以前方法中存在的一些问题,同时保持了较高的"随机性因素",但我不知道是否有任何结果。有吗
- 添加带有的标识int列,并将其用作聚集索引。我在这里的问题是,它是否会对任何事情有所帮助,因为查询无论如何都会使用PK值,即GUID
- 稍微有点像以前的方法,将标识int转换为主键值和聚集索引,并添加一个GUID collmn来进行查询。我看到在2012年的冒险作品中,这是一个选择,但我真的不明白它在哪里有帮助。。。如果我用GUID查询值,聚集索引会有帮助吗?GUID既不是PK,也不是聚集索引
我想这就是我能想到的,我真的倾向于选择COMB方法,并思考一些实验来验证它是否更好以及为什么更好。但在这种情况下,有更好的解决方案吗?
//另外,我必须检索数据库生成的GUID,而不是在客户端创建它。//
((注意,我想把它作为一个注释,但"代码"太大了,不能作为一个简单的注释。但这可能是一个答案。)))
您可以创建客户端(主要是)顺序guid。。。。。作为一名C#开发人员,我更喜欢在数据库之外创建所有关系,然后将所有关系作为Xml发送,让TSQL将其分解,然后在一次数据库命中中重新插入我需要的所有关系(我想偶尔也会删除)。但这种方法也有缺点。我认为int和bigint通常会在表演比赛中"获胜"。。。。。。但是。。。。。。。。大多数时候。。。。。。。。。将guid与客户端的"sequential"结合使用已经足够满足我的需求了。特别是,正如你所建议的。。。您必须在某个时刻移动数据。使用IDENTITY()将PK/FK设置为"阵容"是可行的,但使用GUID(IMHO)更容易。现在,如果我在写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
*/
}
}