SQL Server如何保证可空字段中非空值唯一

数据库 SQL Server
文章阐述了编者同学遇到的SQL Server中一个关于“如何保证可空字段中非空值唯一”的问题,以及编者给出的几种解决方式。

今天同学向我提了一个问题,是SQL Server中一个关于“如何保证可空字段中非空值唯一”的问题,我觉得蛮有意思,现记录下来大家探讨下。

问题是:在一个表里面,有一个允许为空的字段,空是可以重复的,但是不为空的值需要唯一。

表结构如下面代码创建

  1. CREATE TABLE test_tb  
  2. (  
  3.     TestId int not null identity(1,1) primary key,  
  4.     Caption nvarchar(100) null 
  5. );  
  6. GO  
 

#p#

解决方案:

解决方案1:

对于这个问题,大家的第一个想法可能是:在Caption这个字段上面加一个唯一键不就可以了吗?好,我们按着这个思路做下去,先创建唯一索引。

  1. CREATE UNIQUE NONCLUSTERED INDEX un_test_tb   
  2. ON test_tb(Caption)  
  3. GO  

 索引创建好了,我们来测试下效果

  1. INSERT INTO test_tb (Caption)  
  2. VALUES (null)  
  3. GO  
  4. INSERT INTO test_tb (Caption)  
  5. VALUES (null)  
  6. GO  

 运行之后我们会收到下面的错误信息:

  1. 消息 2601,级别 14,状态 1,第 1 行  
  2. 不能在具有唯一索引 'un_test_tb' 的对象 'dbo.test_tb' 中插入重复键的行。  
  3. 语句已终止。  

 所以该解决方案是不行的。

解决方案2:

添加约束,让SQL Server在插入数据的时候,先验证下已有数据中是否有现在要插入的这个值。由于这个约束不是简单的一个运算,因此我们先创建一个函数,然后再在约束中调用这个函数。

创建验证逻辑函数:

  1. CREATE FUNCTION [dbo].[fn_CK_test_tb_Caption]()  
  2. RETURNS BIT 
  3. AS   
  4. BEGIN 
  5.     IF(EXISTS(  
  6.     SELECT    1  
  7.     FROM test_tb AS a  
  8.     WHERE (Caption IS NOT NULLAND EXISTS  
  9.       (SELECT 1 AS Expr1  
  10.         FROM test_tb  
  11.         WHERE (Caption IS NOT NULLAND (Caption = a.Caption) AND (a.TestId <> TestId))  
  12.      ))  
  13.         RETURN 0  
  14.       
  15.     RETURN 1  
  16. END 
  17. GO  

 在约束中引用函数:

  1. ALTER TABLE test_tb  
  2. ADD CONSTRAINT CK_test_tb_Caption CHECK (dbo.fn_CK_test_tb_Caption() = 1)  
  3. GO  
  4.  

 现在来测试下效果。先来测试NULL值
 

  1. INSERT INTO test_tb (Caption)  
  2. VALUES (null)  
  3. GO  
  4. INSERT INTO test_tb (Caption)  
  5. VALUES (null)  
  6. GO  
  7. SELECT * FROM test_tb  
  8. GO  

  可以成功运行,而且也出了多行为NULL的情况。现在再来测试不为空的插入情况。

  1. INSERT INTO test_tb (Caption)  
  2. VALUES (N'AAA')  
  3. GO  
  4. INSERT INTO test_tb (Caption)  
  5. VALUES (N'BBB')  
  6. GO  
  7. INSERT INTO test_tb (Caption)  
  8. VALUES (N'BBB')  
  9. GO  
  10. SELECT * FROM test_tb  
  11. GO  

 结果是在第三条语句的时候报错了,表中的Caption字段也有'AAA'和'BBB'了,这也正好是我们要的结果。

所以解决方案2是正确的。但是为了这么一个小小功能,就写这么长一段东西是不是太繁琐了呢?我们来看下面的解决方案。

  解决方案3:(只适用于SQL Server 2008)

 SQL Server 2008中有了一个优雅的解决方案,那就是筛选索引。筛选索引是一种经过优化的非聚集索引,尤其适用于涵盖从定义完善的数据子集中选择数据的查询。筛选索引使用筛选谓词对表中的部分行进行索引。有了筛选索引,我们只需要写一条语句就达到上面的效果。
 

  1. CREATE UNIQUE NONCLUSTERED INDEX un_test_tb   
  2. ON test_tb(Caption)  
  3. WHERE Caption is not null 
  4. GO  

 再用上面的一些测试语句来测试的话,会发现完全是达到了我们的要求。

这个方案的唯一缺点就是该语句只有SQL Server 2008支持。。

不知道各位有没有又优雅又适用于各个版本的SQL Server的解决方案,望不胜赐教。

【编辑推荐】

  1. SQL Server 2000删除实战演习
  2. SQL Server存储过程的命名标准如何进行?
  3. 卸载SQL Server 2005组件的正确顺序
  4. 对SQL Server字符串数据类型的具体描述
  5. SQL Server数据类型的种类与应用


 

责任编辑:艾婧 来源: 博客园
相关推荐

2010-07-06 10:25:16

SQL Server空

2009-08-25 17:52:01

C#可空值类型

2017-01-13 23:06:45

swiftios

2010-07-02 13:26:56

SQL Server处

2010-09-13 16:39:36

sql server触

2011-04-01 14:14:42

SQL Server空值

2010-11-12 14:39:36

Sql Server唯

2010-11-12 14:45:19

Sql Server唯

2010-10-20 10:04:36

sql server自

2024-05-24 09:29:28

2010-06-28 09:59:47

SQL Server自

2010-09-26 16:39:27

SQL子查询

2021-11-15 06:56:45

系统运行空指针

2020-08-18 22:20:49

vue.jsnullclass

2024-01-10 08:08:25

Python空值校验开发

2010-09-24 17:08:03

SQL Server唯

2010-07-15 10:37:15

SQL Server默

2010-07-09 12:49:41

SQL Server自

2021-07-02 06:54:43

分布式环境ID

2010-09-28 10:35:58

SQL字段默认值
点赞
收藏

51CTO技术栈公众号