LOGO OA教程 ERP教程 模切知识交流 PMS教程 CRM教程 开发文档 其他文档  
 
网站管理员

SQLServer 唯一约束与唯一索引的区别

admin
2025年1月9日 22:1 本文热度 14

唯一约束和唯一索引在表设计中是最常见的,都有唯一属性,用于约束表字段的唯一性。但是,这两者到底有什么区别呢?


接下来我们测试看看,创建两张表分别用于测试唯一约束和唯一索引。

USE [DemoDB]GOCREATE TABLE [dbo].[TableUniqueKey](  id int not null,  name varchar(20) null)GOCREATE TABLE [dbo].[TableUniqueIndex](  id int not null,  name varchar(20) null)GOINSERT INTO [dbo].[TableUniqueKey]SELECT 1,'KK' UNION ALLSELECT 2,NULL UNION ALLSELECT 3,NULL GOINSERT INTO [dbo].[TableUniqueIndex]SELECT 1,'KK' UNION ALLSELECT 2,NULL UNION ALLSELECT 3,NULL GO

现在分别创建唯一键约束和创建唯一索引,发现错误!

--  创建唯一键约束ALTER TABLE [dbo].[TableUniqueKey] ADD CONSTRAINT [IX_TableUniqueKey_name] UNIQUE ([name] ASC)--默认非聚集索引GOALTER TABLE [dbo].[TableUniqueKey] ADD CONSTRAINT [IX_TableUniqueKey_name] UNIQUE NONCLUSTERED([name] ASC)GO--  创建唯一索引CREATE UNIQUE NONCLUSTERED INDEX [IX_UniqueIndex_name] ON [dbo].[TableUniqueIndex]([name] ASC)GO

Msg 1505, Level 16, State 1, Line 1
The CREATE UNIQUE INDEX statement terminated because a duplicate key was found for the object name 'dbo.TableUniqueKey' and the index name 'IX_TableUniqueKey_name'. The duplicate key value is (<NULL>).
Msg 1750, Level 16, State 0, Line 1 Could not create constraint. See previous errors.

The statement has been terminated.

Msg 1505, Level 16, State 1, Line 1
The CREATE UNIQUE INDEX statement terminated because a duplicate key was found for the object name 'dbo.TableUniqueIndex' and the index name 'IX_UniqueIndex_name'. The duplicate key value is (<NULL>).

The statement has been terminated.


可以看到,都提示有重复值,重复值均为 NULL 值。现在删除重复值,再创建唯一约束。

DELETE FROM [dbo].[TableUniqueKey] WHERE ID = 3DELETE FROM [dbo].[TableUniqueIndex] WHERE ID = 3--  创建唯一键约束ALTER TABLE [dbo].[TableUniqueKey] ADD CONSTRAINT [IX_TableUniqueKey_name] UNIQUE NONCLUSTERED([name] ASC)



创建唯一键约束,会同时创建同名的唯一非聚集索引, 以及创建同名统计信息;而唯一键约束是靠唯一索引来约束。


若是对唯一键生成的索引直接删除,则报错!

DROP INDEX [IX_TableUniqueKey_name] ON [dbo].[TableUniqueKey]

Msg 3723, Level 16, State 5, Line 1
An explicit DROP INDEX is not allowed on index 'dbo.TableUniqueKey.IX_TableUniqueKey_name'. It is being used for UNIQUE KEY constraint enforcement.


正确的删除方法,是删除表约束。

ALTER TABLE [dbo].[TableUniqueKey] DROP CONSTRAINT [IX_TableUniqueKey_name]


继续测试,现在对另一张表创建唯一索引。


CREATE UNIQUE NONCLUSTERED INDEX [IX_UniqueIndex_name] ON [dbo].[TableUniqueIndex]([name] ASC)


创建唯一索引, 同时创建同名统计信息。


对索引删除后,我们再重建,对比再看看。

DROP INDEX [IX_UniqueIndex_name] ON [dbo].[TableUniqueIndex]GOALTER TABLE [dbo].[TableUniqueKey] ADD CONSTRAINT [IX_TableUniqueKey_name] UNIQUE NONCLUSTERED([name] ASC)GOCREATE UNIQUE NONCLUSTERED INDEX [IX_UniqueIndex_name] ON [dbo].[TableUniqueIndex]([name] ASC)GOSELECT object_name(object_id),name,index_id,type_desc,is_unique,is_unique_constraintFROM sys.indexes WHERE object_name(object_id) IN ('TableUniqueKey','TableUniqueIndex')


唯一键约束 [TableUniqueKey] 不是 check 约束,是属于一种为 UNIQUE_CONSTRAINT 的约束。而他们的索引都有唯一性约束。


此外,还可以通过以下检查他们的区别:

EXEC sp_helpconstraint 'TableUniqueKey' EXEC sp_helpconstraint 'TableUniqueIndex' 
EXEC sp_helpindex 'TableUniqueKey' EXEC sp_helpindex 'TableUniqueIndex' 
--  EXEC sp_help 'TableUniqueKey' --  EXEC sp_help 'TableUniqueIndex'


对比索引描述中,唯一键 比 唯一 索引多了 unique key 

--唯一键不出错EXEC sp_help [IX_TableUniqueKey_name]EXEC sp_helpindex [IX_TableUniqueKey_name]EXEC sp_helpconstraint [IX_TableUniqueKey_name]--唯一索引出错EXEC sp_help [IX_UniqueIndex_name]EXEC sp_helpindex [IX_UniqueIndex_name]EXEC sp_helpconstraint [IX_UniqueIndex_name]

上面可以看出,唯一不同的是: 唯一键 比 唯一索引 多了一种叫做 "unique key" 的约束。


现在禁用索引,再插入重复数据。

--禁用索引/约束(均可被禁用)ALTER INDEX [IX_TableUniqueKey_name] ON [dbo].[TableUniqueKey] DISABLEALTER INDEX [IX_UniqueIndex_name] ON [dbo].[TableUniqueIndex] DISABLE--插入重复值,正常INSERT INTO [dbo].[TableUniqueKey] SELECT 3,NULL INSERT INTO [dbo].[TableUniqueIndex]SELECT 3,NULL

数据都能正常插入,约束或索引被禁用了。对于唯一键约束,也是禁用索引吗?函数 ObjectProperty() 的参数 CnstIsDisabled 可以确认约束是否被禁用。

SELECT ObjectProperty(object_id('IX_TableUniqueKey_name'),'CnstIsDisabled')

结果为 0 ,即约束没有被禁用,也就是说禁用的是索引,唯一键约束中,唯一性是依赖于其默认创建的唯一索引来约束的!


现在重建索引:

--删除重复DELETE FROM [dbo].[TableUniqueKey] WHERE ID = 3DELETE FROM [dbo].[TableUniqueIndex] WHERE ID = 3/*对索引的更改*/--重建索引ALTER INDEX [IX_TableUniqueKey_name] ON [dbo].[TableUniqueKey] REBUILDALTER INDEX [IX_UniqueIndex_name] ON [dbo].[TableUniqueIndex] REBUILD--更改部分索引参数ALTER INDEX [IX_TableUniqueKey_name] ON [dbo].[TableUniqueKey] SET ( ALLOW_ROW_LOCKS = ON )ALTER INDEX [IX_UniqueIndex_name] ON [dbo].[TableUniqueIndex] SET ( ALLOW_ROW_LOCKS = ON )

两种索引其实还是可以更改一下参数的。使用窗口打开查看,唯一键约束的索引有些是不能更改的。


唯一键约束的索引不能像正常的索引使用太多的索引参数,因为唯一键约束与其索引同在。而单独创建的唯一索引可以设置更多的参数,如 PAD_INDEX, FILLFACTOR, IGNORE_DUP_KEY, DROP_EXISTING, STATISTICS_NORECOMPUTE, and SORT_IN_TEMPDB 。


总的来说,唯一键键约束和唯一索引功能是一样的:"唯一性" + "索引"


唯一键键约束:只是作为一种独特的约束(如主键约束,唯一键约束,check约束,外键约束 的一种),以约束的形式管理.但是同时又自动创建了唯一非聚集索引,也就有了索引的性能和部分功能.实际上唯一键约束是用唯一索引来约束的。


唯一索引:就是一种索引,它对某字段进行唯一性检查,同时可以设置各种参数,非常灵活。


那么我们在创建列的唯一性时,到底使用哪一种较好呢?

唯一键约束在表中是必定存在的约束的,唯一键约束的索引存在于一个分区中,并且不会像索引那样可以更改。因为索引可以随时改动(当然也不会经常改动),个人建议索引还是用唯一索引更灵活。管理约束还得管理索引,而管理索引,一个就好了。但是对于一些高可用性,也要注意索引是否在其他地方也存在。


阅读原文:原文链接


该文章在 2025/1/10 11:06:27 编辑过
关键字查询
相关文章
正在查询...
点晴ERP是一款针对中小制造业的专业生产管理软件系统,系统成熟度和易用性得到了国内大量中小企业的青睐。
点晴PMS码头管理系统主要针对港口码头集装箱与散货日常运作、调度、堆场、车队、财务费用、相关报表等业务管理,结合码头的业务特点,围绕调度、堆场作业而开发的。集技术的先进性、管理的有效性于一体,是物流码头及其他港口类企业的高效ERP管理信息系统。
点晴WMS仓储管理系统提供了货物产品管理,销售管理,采购管理,仓储管理,仓库管理,保质期管理,货位管理,库位管理,生产管理,WMS管理系统,标签打印,条形码,二维码管理,批号管理软件。
点晴免费OA是一款软件和通用服务都免费,不限功能、不限时间、不限用户的免费OA协同办公管理系统。
Copyright 2010-2025 ClickSun All Rights Reserved