背景

  前段时间学习《Microsoft SQL Server 2008技术内幕:T-SQL查询》时,看到里面关于无序GUID作为主键与聚集索引的建议,无序GUID作为主键以及作为聚集索引所带来的问题包括:

  1. 空间的浪费以及由此带来的读写效率的下降。
  2. 更主要的,存储的碎片化(fragmentation)以及由此带来的读写效率严重下降。

所以,尽量避免用GUID(无序或有序)做主键,不要用无序GUID做聚集索引。<摘自博友博客>

  想到在工作中存在一个视图转成物理表的时候使用到了此种场景,分析了一下数据情况,已经有较多客户此表的数据将近百万级,后续会继续线性增长,而且在代码规范也强制要求不允许使用无序GUID,需要调整为有序的GUID,对于修改前后的表现,还是想做一个对比分析...

一、插入无序GUID数据

 --创建表并插入无序GUID数据
--DROP TABLE T_PROORDERTYPEGUIDTEST;
create table T_PROORDERTYPEGUIDTEST (
FENTRYID varchar(36) not null default ' ',
FPROORDERENTRYID int not null default 0,
FPROORDERTYPE varchar(20) not null default ' ',
FFORMID varchar(36) not null default ' ',
FNUMBER nvarchar(160) not null default ' ',
FCREATEORGID int not null default 0,
FUSEORGID int not null default 0,
FDOCUMENTSTATUS char(1) not null default 'C',
FFORBIDSTATUS char(1) not null default 'A',
FDATE datetime not null default getdate(),
FMATERIALID int not null default 0,
FBOMID int not null default 0,
FAUXPROPID int not null default 0,
FLOT int not null default 0,
FMtoNo nvarchar(200) not null default '',
FSEQ int not null default 0,
FUNITID int not null default 0,
FPRODUCTID int not null default 0,
FWORKSHOPID int not null default 0,
FCOSTCENTERID int not null default 0,
constraint PK_PROORDERTYPE primary key (FENTRYID)
)
--插入500000条数据
declare @i int
set @i = 1
while @i < 500000
begin
INSERT INTO T_PROORDERTYPEGUIDTEST(FENTRYID,FPROORDERENTRYID,FPROORDERTYPE,FFORMID,FNUMBER,FCREATEORGID,
FUSEORGID,FDOCUMENTSTATUS,FFORBIDSTATUS,FDATE,FMATERIALID,FBOMID,FAUXPROPID,FLOT,
FSEQ,FUNITID,FPRODUCTID,FWORKSHOPID,FCOSTCENTERID) values
(NEWID(),@i,'PO','FORM_OUTSRCPROORDER','TEST00001',@i,@i,'C','A',GETDATE(),@i,@i,@i,@i,@i,@i,@i,@i,@i)
SET @i=@i+1
end ; select COUNT(1) from T_PROORDERTYPEGUIDTEST

二、插入有序GUID数据

 --创建表并插入有序GUID数据
DROP TABLE T_PROORDERTYPESEQGUIDTEST;
create table T_PROORDERTYPESEQGUIDTEST (
FENTRYID uniqueidentifier not null default (NEWSEQUENTIALID()),
FPROORDERENTRYID int not null default 0,
FPROORDERTYPE varchar(20) not null default ' ',
FFORMID varchar(36) not null default ' ',
FNUMBER nvarchar(160) not null default ' ',
FCREATEORGID int not null default 0,
FUSEORGID int not null default 0,
FDOCUMENTSTATUS char(1) not null default 'C',
FFORBIDSTATUS char(1) not null default 'A',
FDATE datetime not null default getdate(),
FMATERIALID int not null default 0,
FBOMID int not null default 0,
FAUXPROPID int not null default 0,
FLOT int not null default 0,
FMtoNo nvarchar(200) not null default '',
FSEQ int not null default 0,
FUNITID int not null default 0,
FPRODUCTID int not null default 0,
FWORKSHOPID int not null default 0,
FCOSTCENTERID int not null default 0,
constraint PK_SEQPROORDERTYPE primary key (FENTRYID)
)
--插入500000条数据
declare @i int
set @i = 1
while @i < 500000
begin
INSERT INTO T_PROORDERTYPESEQGUIDTEST(FPROORDERENTRYID,FPROORDERTYPE,FFORMID,FNUMBER,FCREATEORGID,
FUSEORGID,FDOCUMENTSTATUS,FFORBIDSTATUS,FDATE,FMATERIALID,FBOMID,FAUXPROPID,FLOT,
FSEQ,FUNITID,FPRODUCTID,FWORKSHOPID,FCOSTCENTERID) values
(@i,'PO','FORM_OUTSRCPROORDER','TEST00001',@i,@i,'C','A',GETDATE(),@i,@i,@i,@i,@i,@i,@i,@i,@i)
SET @i=@i+1
end ; select COUNT(1) from T_PROORDERTYPESEQGUIDTEST

三、分析索引碎片

 --分析索引碎片
declare @table_id int
set @table_id=object_id('T_PROORDERTYPEGUIDTEST')
dbcc showcontig(@table_id);

DBCC SHOWCONTIG 正在扫描 'T_PROORDERTYPEGUIDTEST' 表...
表: 'T_PROORDERTYPEGUIDTEST' (410536596);索引 ID: 1,数据库 ID: 8
已执行 TABLE 级别的扫描。
- 扫描页数................................: 13933
- 扫描区数..............................: 1759
- 区切换次数..............................: 13932
- 每个区的平均页数........................: 7.9
- 扫描密度 [最佳计数:实际计数].......: 12.50% [1742:13933]
- 逻辑扫描碎片 ..................: 99.22%
- 区扫描碎片 ..................: 0.11%
- 每页的平均可用字节数.....................: 2569.6
- 平均页密度(满).....................: 68.25%
DBCC 执行完毕。如果 DBCC 输出了错误信息,请与系统管理员联系。

 declare @table_id int
set @table_id=object_id('T_PROORDERTYPESEQGUIDTEST')
dbcc showcontig(@table_id)

DBCC SHOWCONTIG 正在扫描 'T_PROORDERTYPESEQGUIDTEST' 表...
表: 'T_PROORDERTYPESEQGUIDTEST' (1114539104);索引 ID: 1,数据库 ID: 8
已执行 TABLE 级别的扫描。
- 扫描页数................................: 8197
- 扫描区数..............................: 1033
- 区切换次数..............................: 1032
- 每个区的平均页数........................: 7.9
- 扫描密度 [最佳计数:实际计数].......: 99.23% [1025:1033]
- 逻辑扫描碎片 ..................: 0.67%
- 区扫描碎片 ..................: 0.10%
- 每页的平均可用字节数.....................: 44.3
- 平均页密度(满).....................: 99.45%
DBCC 执行完毕。如果 DBCC 输出了错误信息,请与系统管理员联系。

四、占用空间情况

 sp_spaceused 'T_PROORDERTYPEGUIDTEST';

 sp_spaceused 'T_PROORDERTYPESEQGUIDTEST';

五、查询执行情况

 select * from T_PROORDERTYPEGUIDTEST ;
select * from T_PROORDERTYPESEQGUIDTEST ;

再分别插入50W数据

 declare @i int
set @i = 1
while @i < 500000
begin
INSERT INTO T_PROORDERTYPEGUIDTEST(FENTRYID,FPROORDERENTRYID,FPROORDERTYPE,FFORMID,FNUMBER,FCREATEORGID,
FUSEORGID,FDOCUMENTSTATUS,FFORBIDSTATUS,FDATE,FMATERIALID,FBOMID,FAUXPROPID,FLOT,
FSEQ,FUNITID,FPRODUCTID,FWORKSHOPID,FCOSTCENTERID) values
(NEWID(),@i,'PO','FORM_OUTSRCPROORDER','TEST00001',@i,@i,'C','A',GETDATE(),@i,@i,@i,@i,@i,@i,@i,@i,@i)
SET @i=@i+1
end ; declare @i int
set @i = 1
while @i < 500000
begin
INSERT INTO T_PROORDERTYPESEQGUIDTEST(FPROORDERENTRYID,FPROORDERTYPE,FFORMID,FNUMBER,FCREATEORGID,
FUSEORGID,FDOCUMENTSTATUS,FFORBIDSTATUS,FDATE,FMATERIALID,FBOMID,FAUXPROPID,FLOT,
FSEQ,FUNITID,FPRODUCTID,FWORKSHOPID,FCOSTCENTERID) values
(@i,'PO','FORM_OUTSRCPROORDER','TEST00001',@i,@i,'C','A',GETDATE(),@i,@i,@i,@i,@i,@i,@i,@i,@i)
SET @i=@i+1
end ;

参考文章

http://msdn.microsoft.com/zh-cn/library/ms175008(v=sql.90).aspx

http://msdn.microsoft.com/zh-cn/library/ms188776.aspx

http://www.sqlskills.com/blogs/kimberly/disk-space-is-cheap/

http://www.cnblogs.com/zhouruifu/archive/2012/04/18/2454088.html

Microsoft SQL Server 2008技术内幕:T-SQL查询

[O]SQL SERVER下有序GUID和无序GUID作为主键&聚集索引的性能表现的更多相关文章

  1. SQL SERVER下有序GUID和无序GUID作为主键&聚集索引的性能表现

     背景 前段时间学习<Microsoft SQL Server 2008技术内幕:T-SQL查询>时,看到里面关于无序GUID作为主键与聚集索引的建议,无序GUID作为主键以及作为聚集索引 ...

  2. SQL Server 2014新特性探秘(3)-可更新列存储聚集索引

    简介      列存储索引其实在在SQL Server 2012中就已经存在,但SQL Server 2012中只允许建立非聚集列索引,这意味着列索引是在原有的行存储索引之上的引用了底层的数据,因此会 ...

  3. SQL Server下实现利用SQL Server Agent Job对索引重建实现Balance Load

    昨天工作中遇到这样一个场景,有个项目需要把某台服务器下所有的表和索引都启用数据压缩(data_compression=page),已经启用了的表和索引就不需要再压缩一次了.统计一下后发现要运行的REB ...

  4. Microsoft SQL Server下的SQL语句

    SQL语句先前写的时候,很容易把一些特殊的用法忘记,我特此整理了一下SQL语句操作.一.基础 1.说明:创建数据库CREATE DATABASE database-name 2.说明:删除数据库dro ...

  5. ACCESS与SQL Server下SQL Like 查询的不同

    在ACCESS中LIKE的用法Access里like的通配符用法是这样: “?”表示任何单一字符: “*”表示零个或多个字符: “#”表示任何一个数字 所以应该是: select * from dat ...

  6. SQL Server下ADO.NET 怎么获取数据库SQL语句INSERT,UPDATE,DELETE了多少行数据

    ADO.NET 在发送SQL语句到SQL Server数据库后,怎么知道真正INSERT,UPDATE,DELETE了多少行数据呢? 使用SQL Server内置的全局变量@@ROWCOUNT即可,@ ...

  7. SQL SERVER中用户定义标量函数(scalar user defined function)的性能问题

    用户定义函数(UDF)分类  SQL SERVER中的用户定义函数(User Defined Functions 简称UDF)分为标量函数(Scalar-Valued Function)和表值函数(T ...

  8. Sql Server Analysis Service 处理时找到重复的属性键、找不到属性键错误(转载)

    这是两个非常常见的SSAS处理异常,网上也能找到很多文章讲解决办法,但很少见关于异常原因的分析,先来看看第一个" OLAP 存储引擎中存在错误: 处理时找到重复的属性键",一个维度 ...

  9. 三、Sql Server 基础培训《进度3-是否使用外键(知识点学习)》

    学习作业3: 问题1:你觉得外键有哪些适用情况?哪些不适用情况?   问题2:本次实战案例,由你来架构,你觉得有必要建立外键吗? 说明你的理由?     ======================= ...

随机推荐

  1. Java面向对象 异常

     Java面向对象  异常 知识概要:                  (1)异常的概述                  (2)异常的体系                  (3)异常的处理 ...

  2. 前端页面卡顿、也许是DOM操作惹的祸?

    界面上UI的更改都是通过DOM操作实现的,并不是通过传统的刷新页面实现 的.尽管DOM提供了丰富接口供外部调用,但DOM操作的代价很高,页面前端代码的性能瓶颈也大多集中在DOM操作上,所以前端性能优化 ...

  3. css笔记-文本样式

    聊聊text-decoration.text-indent.text-transform.letter-spacing.word-spacing.vertical-align.下面是一些常用设置文本样 ...

  4. OpenWRT 恢复出厂设置命令

    如果通过无线或者有线口无法连接到router,可以用恢复某些设置重新设置路由器. 1. 开机,等着一个工作灯亮的时候立即按下rest键2秒,然后就开始拼命闪烁,很好现在进入failsafe模式了. 2 ...

  5. java基础---字符串string

    1.字符创的概念 java字符串就是Unicode字符序列.例如,串“Java\u2122”由5个Unicode字符J.a.v.a和TM.java没有内置的字符串类型,而是在标准库Java类库中提供了 ...

  6. Spring批量更新batchUpdate提交和Hibernate批量更新executeUpdate

    1:先看hibernate的批量更新处理. 版本背景:hibernate 5.0.8 applicationContext.xml 配置清单: <?xml version="1.0&q ...

  7. daterangepicker 使用方法以及各种小bug修复

    双日历时间段选择插件 — daterangepicker是bootstrap框架后期的一个时间控件,可以设定多个时间段选项,也可以自定义时间段,由用户自己选择起始时间和终止时间,时间段的最大跨度可以在 ...

  8. HTTP认证方式与https简介

    HTTP认证与https简介 HTTP请求报头: Authorization [ˌɔ:θəraɪˈzeɪʃn] HTTP响应报头: WWW-Authenticate [ɔ:ˈθentɪkeɪt] HT ...

  9. jvm - 垃圾回收

    jvm - 垃圾回收 注意 : 本系列文章为学习系列,部分内容会取自相关书籍或者网络资源,在文章中间和末尾处会有标注 垃圾回收的意义 它使得java程序员不再时时刻刻的关注内存管理方面的工作. 垃圾回 ...

  10. RSA加密算法验证(C#实现)

    RSA算法简单原理介绍(节选于网络) 假设Alice想要通过一个不可靠的媒体接收Bob的一条私人讯息.她可以用以下的方式来产生一个公钥和一个私钥: 随意选择两个大的质数p和q,p不等于q,计算N=pq ...