触发器是一种特殊的存储过程,它的执行不是由程序调用,也不是手动执行,而是由事件来触发。触发器是当对某一个表进行操做。例如:update、insert、delete这些操做的时候,系统会自动调用执行该表上对应的触发器。sql
一、DML( 数据操纵语言 Data Manipulation Language)触发器:是指触发器在数据库中发生 DML 事件时将启用。DML事件是指在表或视图中对数据进行的 insert、update、delete 操做的语句。数据库
二、DDL(数据定义语言 Data Definition Language)触发器:是指当服务器或数据库中发生 DDL 事件时将启用。DDL事件是指在表或索引中的 create、alter、drop 操做语句。安全
三、登录触发器:是指当用户登陆 SQL SERVER 实例创建会话时触发。若是身份验证失败,登陆触发器不会触发。服务器
其中 DML 触发器比较经常使用,根据 DML 触发器触发的方式不一样又分为如下两种状况:测试
after 触发器(以后触发):其中 after 触发器要求只有执行 insert、update、delete 某一操做以后触发器才会被触发,且只能定义在表上。加密
instead of 触发器 (以前触发):instead of 触发器并不执行其定义的操做(insert、update、delete)而仅是执行触发器自己。能够在表或视图上定义 instead of 触发器。3d
AFTER 触发器语法:对象
CREATE [ OR ALTER ] TRIGGER [ schema_name . ]trigger_name ON { table } [ WITH <dml_trigger_option> [ ,...n ] ] { FOR | AFTER } { [ INSERT ] [ , ] [ UPDATE ] [ , ] [ DELETE ] } AS { sql_statement [ ; ] [ ,...n ] } <dml_trigger_option> ::= [ NATIVE_COMPILATION ] [ SCHEMABINDING ] [ EXECUTE AS Clause ]
INSTEAD OF 触发器语法:blog
CREATE [ OR ALTER ] TRIGGER [ schema_name . ]trigger_name ON { table | view } [ WITH <dml_trigger_option> [ ,...n ] ] { FOR | AFTER | INSTEAD OF } { [ INSERT ] [ , ] [ UPDATE ] [ , ] [ DELETE ] } [ WITH APPEND ] [ 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
DDL 触发器语法:索引
CREATE [ OR ALTER ] TRIGGER trigger_name ON { ALL SERVER | DATABASE } [ WITH <ddl_trigger_option> [ ,...n ] ] { FOR | AFTER } { event_type | event_group } [ ,...n ] AS { sql_statement [ ; ] [ ,...n ] | EXTERNAL NAME < method specifier > [ ; ] } <ddl_trigger_option> ::= [ ENCRYPTION ] [ EXECUTE AS Clause ]
登录触发器语法:
CREATE [ OR 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 ]
参数介绍:
1. 建立AFTER INSERT 触发器:当下单后更新tbOrderTotalPrice 的统计信息
准备一张订单表(tbOrder),包含订单ID,订单交易金额,订单建立时间,准备订单交易分段时间总额(tbOrderTotalPrice ),用来统计每月的交易合计金额
USE [TEST] GO /****** Object: Trigger [dbo].[tSumTotalOrderPrice] Script Date: 2018/5/15 11:10:04 ******/ SET ANSI_NULLS ON GO SET QUOTED_IDENTIFIER ON GO CREATE TRIGGER [dbo].[tSumTotalOrderPrice] on [dbo].[tbOrder] AFTER INSERT AS BEGIN DECLARE @DT INT SET @DT= CONVERT(varchar(6),GETDATE(),112) DECLARE @SUM DECIMAL(18, 2) SET @SUM=(SELECT SUM(OPrice) FROM tbOrder WHERE @DT = CONVERT(varchar(6),CreateDT,112)) IF(EXISTS(SELECT * FROM tbOrderTotalPrice WHERE YearMoth=@DT)) BEGIN UPDATE tbOrderTotalPrice SET TotalPrice=@SUM WHERE YearMoth=@DT END ELSE BEGIN INSERT INTO tbOrderTotalPrice VALUES (@DT,@SUM) END END GO
执行INSERT 行为:INSERT INTO [dbo].[tbOrder]([OPrice],[CreateDT])VALUES(9,GETDATE())
2.建立 instead of insert 触发器,当添加用户成绩记录时候,验证学号和学科存在才能进行添加操做
准备学生表(tbStudent),学科表(tbSubject),成绩表(tbStuSubScore)
在学生表中添加测试学生,在学科标准添加测试学科
新增instead of insert 触发器,在执行INSERT 语句前验证学号和学科存在才能进行添加操做
USE [TEST] GO /****** Object: Trigger [dbo].[tCheckStuSubScore] Script Date: 2018/5/17 14:20:53 ******/ SET ANSI_NULLS ON GO SET QUOTED_IDENTIFIER ON GO -- ============================================= -- Author: yangyi -- Create date: 2018/5/17 -- Description: 新增学科分数 -- ============================================= CREATE TRIGGER [dbo].[tCheckStuSubScore] ON [dbo].[tbStuSubScore] INSTEAD OF INSERT AS BEGIN --验证学号是否存在 IF(NOT EXISTS(SELECT * FROM tbStudent WHERE StuID=(SELECT StuID FROM inserted))) BEGIN PRINT('验证学号不否存在') ROLLBACK TRANSACTION END ELSE BEGIN --验证学科是否存在 IF(NOT EXISTS(SELECT * FROM tbSubject WHERE SubID=(SELECT SubID FROM inserted))) BEGIN PRINT('验证学科不否存在') ROLLBACK TRANSACTION END ELSE BEGIN --执行INSERT INSERT INTO tbStuSubScore SELECT * FROM inserted END END END GO
注意:
在触发器语句中用两个特殊的表一个是deleted表和inserted。它们是经过触发器操做自动建立驻留在内存中的临时表。
Deleted表用于存储 DELETE和 UPDATE语句所影响的行的复本。在执行DELETE或 UPDATE语句时,行从触发器表中删除,并传输到 deleted表中。Deleted表和触发器表一般没有相同的行。
Inserted 表用于存储 INSERT 和 UPDATE 语句所影响的行的副本。在一个插入或更新事务处理中,新建行被同时添加到 inserted 表和触发器表中。Inserted 表中的行是触发器表中新行的副本
1.插入操做(Insert) Inserted表有数据,Deleted表无数据
2.删除操做(Delete) Inserted表无数据,Deleted表有数据
3.更新操做(Update) Inserted表有数据(新数据),Deleted表有数据(旧数据)
测试:
1>.执行INSERT 添加一条学科不存不正常在的学生成绩记录,执行失败
INSERT INTO [dbo].[tbStuSubScore]([StuID],[Score],[SubID],[CreateDT])VALUES('001',100,0,GETDATE())
2.>执行INSERT 添加一条学号不存不正常在的学生成绩记录,执行失败
INSERT INTO [dbo].[tbStuSubScore]([StuID],[Score],[SubID],[CreateDT])VALUES('004',100,1,GETDATE())
3.>执行一条正常的学生成绩记录,执行成功
INSERT INTO [dbo].[tbStuSubScore]([StuID],[Score],[SubID],[CreateDT])VALUES('001',100,1,GETDATE())