我们有一个用户表,每个用户都有一个唯一的电子邮件和用户名。我们尝试在我们的代码中执行此操作,但我们希望确保用户永远不会使用相同的电子邮件用户名插入(或更新)数据库。
我添加了一个BEFORE INSERT
触发器,可以阻止重复用户的插入。
CREATE TRIGGER [dbo].[BeforeUpdateUser]
ON [dbo].[Users]
INSTEAD OF INSERT
AS
BEGIN
-- SET NOCOUNT ON added to prevent extra result sets from
-- interfering with SELECT statements.
SET NOCOUNT ON;
DECLARE @Email nvarchar(MAX)
DECLARE @UserName nvarchar(MAX)
DECLARE @UserId int
DECLARE @DoInsert bit
SET @DoInsert = 1
SELECT @Email = Email, @UserName = UserName FROM INSERTED
SELECT @UserId = UserId FROM Users WHERE Email = @Email
IF (@UserId IS NOT NULL)
BEGIN
SET @DoInsert = 0
END
SELECT @UserId = UserId FROM Users WHERE UserName = @UserName
IF (@UserId IS NOT NULL)
BEGIN
SET @DoInsert = 0
END
IF (@DoInsert = 1)
BEGIN
INSERT INTO Users
SELECT
FirstName,
LastName,
Email,
Password,
UserName,
LanguageId,
Data,
IsDeleted
FROM INSERTED
END
ELSE
BEGIN
DECLARE @ErrorMessage nvarchar(MAX)
SET @ErrorMessage =
'The username and emailadress of a user must be unique!'
RAISERROR 50001 @ErrorMessage
END
END
但是对于Update触发器,我不知道如何做到这一点。 我用谷歌找到了这个例子: http://www.devarticles.com/c/a/SQL-Server/Using-Triggers-In-MS-SQL-Server/2/ 但是,当您一次更新多个列时,我不知道它是否适用。
修改
我尝试在这些列上添加一个唯一约束,但它不起作用:
Msg 1919, Level 16, State 1, Line 1
Column 'Email' in table 'Users' is of a type
that is invalid for use as a key column in an index.
答案 0 :(得分:12)
您可以在表格上添加唯一约束,如果您尝试插入或更新并创建重复项,则会引发错误
ALTER TABLE [Users] ADD CONSTRAINT [IX_UniqueUserEmail] UNIQUE NONCLUSTERED
(
[Email] ASC
)
ALTER TABLE [Users] ADD CONSTRAINT [IX_UniqueUserName] UNIQUE NONCLUSTERED
(
[UserName] ASC
)
编辑:好的,我刚刚在另一篇文章中看到了您的评论,并发现您使用的是NVARCHAR(MAX)作为您的数据类型。您是否有理由要求超过4000个字符的电子邮件地址或用户名?这就是你的问题所在。如果将其减少到NVARCHAR(250)或其左右,则可以使用唯一索引。
答案 1 :(得分:4)
听起来很多工作,而不仅仅是使用一个或多个唯一索引。你有没有走过索引路线的原因?
答案 2 :(得分:1)
为什么不在数据库中的列上使用UNIQUE属性?设置将使SQL服务器强制执行该操作并在尝试插入欺骗时抛出错误。
答案 3 :(得分:1)
您应该对每个列使用SQL UNIQUE
约束。
答案 4 :(得分:0)
您可以UNIQUE INDEX
创建NVARCHAR
,只要NVARCHAR(450)
或更短。{/ p>
你真的需要UNIQUE
列这么大吗?
答案 5 :(得分:0)
一般情况下,我会尽可能避免使用触发器,因为除非您知道触发器存在,否则它们会使行为很难理解。正如其他评论员所说,一个独特的约束是要走的路(一旦你修改了你的列定义以允许它)。
如果您发现自己需要使用触发器,则可能表明您的设计存在缺陷。仔细想想你为什么需要它以及它是否正在执行属于别处的逻辑。
答案 6 :(得分:0)
请注意,如果对SQL Server使用UNIQUE约束/索引解决方案,则该列中只允许一个空值。因此,例如,如果您希望电子邮件地址是可选的,则无法使用,因为只有一个用户可以使用空电子邮件地址。在这种情况下,您将不得不采用另一种方法,如触发器或过滤索引。