由列位控制的UNIQUE约束

时间:2010-10-22 14:33:29

标签: sql sql-server database-design unique-constraint

我有一张桌子,比如

FieldsOnForms(
 FieldID int (FK_Fields)
 FormID int (FK_Forms)
 isDeleted bit
)

该对(FieldID,FormID)应该是唯一的,但仅当未删除该行时(isDeleted = 0)。

是否可以在SQLServer 2008中定义这样的约束? (不使用触发器)

P.S。将(FieldID,FormID,isDeleted)设置为唯一添加了将一行标记为已删除的可能性,但我希望有机会将n行(每个FieldID,FormID)设置为isDeleted = 1,并且只有一行具有isDeleted = 0

3 个答案:

答案 0 :(得分:13)

您可以使用SQL Server 2008 filtered indexes功能获得唯一的索引,或者您可以对视图应用UNIQUE索引(穷人的过滤索引,适用于早期版本) ,但你不能有像你所描述的那样的UNIQUE约束。

过滤索引的示例:

 CREATE UNIQUE NONCLUSTERED INDEX IX_FieldsOnForms_NonDeletedUnique ON FieldsOnForms (FieldID,FormID) WHERE isDeleted=0

答案 1 :(得分:3)

您可以将IsDeleted列更改为DeletedDate,并将其设置为DATETIME,其中包含逻辑删除行的确切时间。或者,您可以添加DeletedDate列,然后针对该列创建一个IsDeleted计算列,以便在代码中使用该列时仍然可以使用该列。然后,您当然会在DeletedDate上添加唯一索引(除了FieldID和FormId)而不是IsDeleted列。这将只允许一个NULL列。

Albin发布了类似的解决方案,但后来删除了它。我不知道为什么,但如果他重新发帖,那么他就在我的面前。 :)

答案 2 :(得分:1)

不,独特意味着真正独一无二。您必须将已删除的记录移动到另一个表或将IsDeleted更改为在所有已删除的记录(例如时间戳)中可以是唯一的。这两种解决方案都需要在您的应用程序,存储过程或触发器中进行额外的工作。