ALTER TRIGGER (Transact-SQL)

修改 CREATE TRIGGER 语句以前创建的 DML、DDL 或登录触发器的定义。触发器是通过使用 CREATE TRIGGER 创建的。这些触发器可以由 Transact-SQL 语句直接创建,也可以由程序集方法创建,这些方法在 Microsoft .NET Framework 公共语言运行时 (CLR) 中创建并上载到 SQL Server 的实例。有关 ALTER TRIGGER 语句中使用的参数的详细信息,请参阅CREATE TRIGGER (Transact-SQL)

主题链接图标Transact-SQL 语法约定

语法

Trigger on an INSERT, UPDATE, or DELETE statement to a table or view (DML Trigger)
ALTER TRIGGER schema_name.trigger_name 
ON ( table | view ) 
[ WITH <dml_trigger_option> [ ,...n ] ]
(FOR | AFTER | INSTEAD OF ) 
{ [ DELETE ] [ , ] [ INSERT ] [ , ] [ UPDATE ] } 
[ NOT FOR REPLICATION ] 
AS { sql_statement [ ; ] [ ...n ] | EXTERNAL NAME <method specifier> [ ; ] } 

<dml_trigger_option> ::=
    [ ENCRYPTION ]
    [ <EXECUTE AS Clause> ]

<method_specifier> ::=
        assembly_name.class_name.method_name

Trigger on a CREATE, ALTER, DROP, GRANT, DENY, REVOKE, or UPDATE statement (DDL Trigger)

ALTER TRIGGER trigger_name 
ON { DATABASE | ALL SERVER } 
[ WITH <ddl_trigger_option> [ ,...n ] ]
{ FOR | AFTER } { event_type [ ,...n ] | event_group } 
AS { sql_statement [ ; ] | EXTERNAL NAME <method specifier> 
[ ; ] }
} 

<ddl_trigger_option> ::=
    [ ENCRYPTION ]
    [ <EXECUTE AS Clause> ]

<method_specifier> ::=
        assembly_name.class_name.method_name

Trigger on a LOGON event (Logon Trigger)
ALTER TRIGGER trigger_name 
ON ALL SERVER 
[ WITH <logon_trigger_option> [ ,...n ] ]
{ FOR| AFTER } LOGON 
AS { sql_statement  [ ; ] [ ,...n ] | EXTERNAL NAME < method specifier >  [ ; ] }

<logon_trigger_option> ::=
    [ ENCRYPTION ]
    [ EXECUTE AS Clause ]

<method_specifier> ::=
    assembly_name.class_name.method_name

参数

  • schema_name
    DML 触发器所属架构的名称。DML 触发器的作用域是为其创建该触发器的表或视图的架构。schema_name 仅在 DML 触发器及其对应的表或视图属于默认架构时可选。对于 DDL 或登录触发器,无法指定 schema_name。

  • trigger_name
    要修改的现有触发器。

  • table | view
    对其执行 DML 触发器的表或视图。可以选择指定表或视图的完全限定名称。

  • DATABASE
    将 DDL 触发器的作用域应用于当前数据库。如果指定了此参数,则只要当前数据库中出现 event_type 或 event_group,就会激发该触发器。

  • ALL SERVER
    将 DDL 或登录触发器的作用域应用于当前服务器。如果指定了此参数,则只要当前服务器中的任何位置上出现 event_type 或 event_group,就会激发该触发器。

  • WITH ENCRYPTION
    加密包含 ALTER TRIGGER 语句文本的 sys.syscomments 和 sys.sql_modules 项。使用 WITH ENCRYPTION 可以防止将触发器作为 SQL Server 复制的一部分进行发布。不能为 CLR 触发器指定 WITH ENCRYPTION。

    注意注意

    如果触发器是使用 WITH ENCRYPTION 创建的,则要使该选项保持启用,必须在 ALTER TRIGGER 语句中再次指定。

  • EXECUTE AS
    指定用于执行该触发器的安全上下文。让您能够控制 SQL Server 的实例在验证对触发器引用的任何数据库对象的权限时使用的用户帐户。

    有关详细信息,请参阅 EXECUTE AS 子句 (Transact-SQL)

  • AFTER
    指定只有在触发 SQL 语句成功执行后,才会激发触发器。所有的引用级联操作和约束检查都成功完成后,才能激发此触发器。

    如果仅指定了 FOR 关键字,那么 AFTER 是默认设置。

    只能对表定义 DML AFTER 触发器。

  • INSTEAD OF
    指定执行 DML 触发器而不是触发 SQL 语句,因此,其优先级高于触发语句的操作。不能为 DDL 或登录触发器指定 INSTEAD OF。

    对于表或视图,每个 INSERT、UPDATE 或 DELETE 语句最多可定义一个 INSTEAD OF 触发器。但是,可以为具有自己的 INSTEAD OF 触发器的多个视图定义视图。

    不允许在使用 WITH CHECK OPTION 创建的视图上定义 INSTEAD OF 触发器。将 INSTEAD OF 触发器添加到指定了 WITH CHECK OPTION 的视图时,SQL Server 将引发错误。用户必须用 ALTER VIEW 删除该选项后才能定义 INSTEAD OF 触发器。

  • { [DELETE] [,] [INSERT] [,] [UPDATE] } | { [INSERT] [,] [UPDATE]}
    指定数据修改语句在试图修改表或视图时,激活 DML 触发器。必须至少指定一个选项。在触发器定义中允许使用以任意顺序组合的这些选项。如果指定的选项多于一个,需用逗号分隔这些选项。

    对于 INSTEAD OF 触发器,不允许对具有指定级联操作 ON DELETE 的引用关系的表使用 DELETE 选项。同样,也不允许对具有指定级联操作 ON UPDATE 的引用关系的表使用 UPDATE 选项。有关详细信息,请参阅 ALTER TABLE (Transact-SQL)

  • event_type
    执行之后将导致激发 DDL 触发器的 Transact-SQL 语言事件的名称。DDL 事件中列出了 DDL 触发器的有效事件。

  • event_group
    预定义的 Transact-SQL 语言事件分组的名称。执行任何属于 event_group 的 Transact-SQL 语言事件之后,都将激发 DDL 触发器。DDL 事件组中列出了 DDL 触发器的有效事件组。在 ALTER TRIGGER 运行完成后,event_group 还将充当宏,将它涉及的事件类型添加到 sys.trigger_events 目录视图中。

  • NOT FOR REPLICATION
    表示当复制代理修改触发器所涉及的表时,不应执行该触发器。有关详细信息,请参阅使用 NOT FOR REPLICATION 来控制约束、标识和触发器

  • sql_statement
    触发条件和操作。

  • <method_specifier>
    指定要与触发器绑定的程序集的方法。该方法不能带有任何参数,并且必须返回空值。class_name 必须是有效的 SQL Server 标识符,并且该类必须存在于可见程序集中。该类不能为嵌套类。

注释

有关 ALTER TRIGGER 的详细信息,请参阅 CREATE TRIGGER (Transact-SQL) 中的“备注”部分。

DML 触发器

通过表和视图上的 INSTEAD OF 触发器,ALTER TRIGGER 支持可手动更新的视图。SQL Server 以相同的方式对所有类型的触发器(AFTER、INSTEAD-OF)应用 ALTER TRIGGER。

可以使用 sp_settriggerorder 来指定要对表执行的第一个和最后一个 AFTER 触发器。对一个表只能指定第一个和最后一个 AFTER 触发器。如果在同一个表上还有其他 AFTER 触发器,这些触发器将随机执行。

如果 ALTER TRIGGER 语句更改了第一个或最后一个触发器,将删除所修改触发器上设置的第一个或最后一个属性,并且必须使用 sp_settriggerorder 重置顺序值。

只有在成功执行触发 SQL 语句之后,才会执行 AFTER 触发器。判断执行成功的标准是:执行了所有与已更新对象或已删除对象相关联的引用级联操作和约束检查。AFTER 触发器操作要检查触发语句的效果,也包括所有由触发语句引起的 UPDATE 和 DELETE 引用级联操作。

如果一个子表或引用表上的 DELETE 操作是由于父表的 CASCADE DELETE 操作所引起的,并且子表上定义了 DELETE 的 INSTEAD OF 触发器,那么将忽略该触发器并执行 DELETE 操作。

DDL 触发器

与 DML 触发器不同,DDL 触发器的作用域不是架构。因此,在查询有关 DDL 触发器的元数据时,不能使用 OBJECT_ID、OBJECT_NAME、OBJECTPROPERTY 和 OBJECTPROPERTY(EX)。请改用目录视图。有关详细信息,请参阅获取有关 DDL 触发器的信息

权限

若要更改 DML 触发器,需要对于定义该触发器所在的表或视图拥有 ALTER 权限。

若要更改定义了服务器范围 (ON ALL SERVER) 的 DDL 触发器或者更改登录触发器,需要对该服务器拥有 CONTROL SERVER 权限。若要更改定义了数据库范围 (ON DATABASE) 的 DDL 触发器,需要对当前数据库拥有 ALTER ANY DATABASE DDL TRIGGER 权限。

示例

以下示例创建一个 DML 触发器,当用户要在 SalesPersonQuotaHistory 表中添加或更改数据时,该触发器将把用户定义的消息打印到客户端。然后使用 ALTER TRIGGER 对该触发器进行了修改,以便只将其应用于 INSERT 活动。该触发器有助于提醒向表中插入行或更新行的用户及时通知 Compensation 部门。

USE AdventureWorks2008R2;
GO
IF OBJECT_ID(N'Sales.bonus_reminder', N'TR') IS NOT NULL
    DROP TRIGGER Sales.bonus_reminder;
GO
CREATE TRIGGER Sales.bonus_reminder
ON Sales.SalesPersonQuotaHistory
WITH ENCRYPTION
AFTER INSERT, UPDATE 
AS RAISERROR ('Notify Compensation', 16, 10);
GO
-- Now, change the trigger.
USE AdventureWorks2008R2;
GO
ALTER TRIGGER Sales.bonus_reminder
ON Sales.SalesPersonQuotaHistory
AFTER INSERT
AS RAISERROR ('Notify Compensation', 16, 10);
GO