-- 先模拟环境,后面说明:
USE [Temp]
GO
-- DROP TABLE [TestTab] TRUNCATE TABLE [TestTab]
CREATE TABLE [dbo].[TestTab](
[UserAcount] [varchar](50) NOT NULL,
[UserName] [varchar](50) NOT NULL,
[crdatetime] [datetime] NOT NULL,
[value] [numeric](18, 4) NULL,
[Info] [varchar](50) NULL,
CONSTRAINT [PK_TestTab] PRIMARY KEY CLUSTERED ([UserAcount] ASC,[UserName] ASC)
) ON [PRIMARY]
GO
-- DROP VIEW [VTestTab]
CREATE VIEW [dbo].[VTestTab]
AS
SELECT [UserAcount],[UserName],[crdatetime],[value],[Info]
FROM [dbo].[TestTab]
WHERE [UserAcount] = CURRENT_USER
GO
-- 插入测试数据: 2538 行
INSERT INTO [TestTab]([UserAcount],[UserName],[crdatetime],[value],[Info])
SELECT CURRENT_USER,name,MAX(crdate),FLOOR(RAND(ABS(CHECKSUM(NEWID())))*1000),NULL
FROM master.sys.sysobjects
WHERE LEN(name)>1 AND LEN(name)<50
GROUP BY name
实际环境:
上面插入的数据是一个用户(CURRENT_USER)的数据,表中还存在更多用户.
由于是按用户划分管理(分库分区等)的,所以列名称[UserAcount]作为聚集索引首先放在第一列
因此,用户都是访问视图[VTestTab],查看到的只有用户自己的数据.
现有以下这个查询:
SELECT * FROM [VTestTab]
WHERE [crdatetime] = '2011-06-17 03:18:08.647'
<??http://www.2cto.com/kf/ware/vc/" target="_blank" class="keylink">vcD4KPHA+PHN0cm9uZz6/tNa00NC8xruuLLLp0a/Kx9ffy/fS/bXELtLyzqrK0828vavXqrHkzqq21LHttcSy6dGvLMjnz8I6PC9zdHJvbmc+PGJyPgo8L3A+CjxwPjwvcD4KPHByZSBjbGFzcz0="brush:sql;">SELECT * FROM [TestTab] WHERE [UserAcount] = CURRENT_USER AND [crdatetime] = '2011-06-17 03:18:08.647'
开始以为,这个使用聚集索引已无法优化了.但是想想,按聚集索引的原理,使用聚集索引查找,表中[UserAcount] = CURRENT_USER 的数据将全部符合,也就是聚集索引将会把当前用户的数据全部查询一遍,即按主键列[UserAcount]查找一遍,并没有准确定位到时间点'2011-06-17 03:18:08.647'这行
所以,现在测试看看到底聚集索引到底查询了多少行!
--使用序列化查看,在事务结束前查看锁情况。
SET TRANSACTION ISOLATION LEVEL SERIALIZABLE
BEGIN TRAN
SELECT * FROM [VTestTab]
WHERE [crdatetime] = '2011-06-17 03:18:08.647'
select resource_type,resource_description,request_mode,request_status,request_type,request_lifetime
from sys.dm_tran_locks where resource_database_id=DB_ID() and request_session_id=@@SPID
COMMIT TRAN
可以看到,整个哈希键都查找了一遍.有页锁(PAGE:IS),键范围锁(KEY:RangeS-S)。把符合索引键[UserAcount]的用户到查询出来了!但最终只返回一行。
如果表中的数据达到了锁升级要求(表行数五六千以上),对表将升级为共享锁!

既然不是最好的,那就考虑另外的索引定位了!
当前聚集索引键列为:([UserAcount],[UserName])
现考虑4种索引创建方法,到底哪种比较较好!~
-- DROP INDEX IX_TestTab ON [TestTab]
CREATE NONCLUSTERED INDEX IX_TestTab ON [TestTab]([UserAcount],[crdatetime])
CREATE NONCLUSTERED INDEX IX_TestTab ON [TestTab]([crdatetime])
CREATE NONCLUSTERED INDEX IX_TestTab ON [TestTab]([crdatetime],[UserAcount])
CREATE NONCLUSTERED INDEX IX_TestTab ON [TestTab]([crdatetime])INCLUDE([value],[Info])
--第一种:
CREATE NONCLUSTERED INDEX IX_TestTab ON [TestTab]([UserAcount],[crdatetime])
这种还是使用聚集索引,并没有用到新的索引,因为新的索引第一个键列为 [UserAcount],与聚集索引一样,所以只要有[UserAcount] ,查询就使用聚集索引了!这个索引加上去也是白加。
--第二种:
CREATE NONCLUSTERED INDEX IX_TestTab ON [TestTab]([crdatetime])
这个查询使用了索引查找,直接按列 [crdatetime] 查询,但是其他不包含在索引的列,使用了键查找。还得在索引子页中查找非索引列。并且锁少了很多!!
--第三种:
CREATE NONCLUSTERED INDEX IX_TestTab ON [TestTab]([crdatetime],[UserAcount])
这种查询的执行计划与上面的一致,因为缓存计划进行了参数化,两者执行语句一样,使用的索引不影响执行计划。而且锁定的资源几乎一样。这个索引的另一列 [UserAcount] 其实是多余了,因为非聚集索引中都会包含聚集索引的键列。所以这个索引的第二列[UserAcount]可以去掉了。
--第四种:
CREATE NONCLUSTERED INDEX IX_TestTab ON [TestTab]([crdatetime])INCLUDE([value],[Info])
这个锁定的资源就比较直接了!~完全使用非聚集索引查找,查询直接定位到数据行!~
所以对于聚集索引,应尽量使用唯一列作为聚集索引,或者最为键列的数据尽量不要重复,这样才能以最快速度定位到行。若没有唯一列,像上面的例子中,聚集索引和另一个比较有效的列作为组合索引聚集索引!~