在实际开发中,我们经常会遇到这样的情况:有 2 个或者多个相互关联的表,如 商品信息
和 库存信息
分别存放在 2 个不同的数据表中,我们在添加一条新商品记录的时候,为了保证数据的完整性,必须同时
在库存表中添加一条库存记录。
这样一来,我们就必须把这两个关联的操作步骤写到程序里面,而且要用 事务
包裹起来,确保这两个操
作成为一个 原子操作
,要么全部执行,要么全部不执行。要是遇到特殊情况,可能还需要对数据进行手
动维护,这样就很 容易忘记其中的一步
,导致数据缺失。
这个时候,咱们可以使用触发器。**你可以创建一个触发器,让商品信息数据的插入操作自动触发库存数 据的插入操作**
。这样一来,就不用担心因为忘记添加库存数据而导致的数据缺失了。
MySQL从 5.0.2 版本开始支持触发器。MySQL的触发器和存储过程一样,都是嵌入到MySQL服务器的一段程序。
触发器是由 事件来触发 某个操作,这些事件包括 INSERT 、 UPDATE 、 DELETE 事件。所谓事件就是指用户的动作或者触发某项行为。如果定义了触发程序,当数据库执行这些语句时候,就相当于事件发生了,就会 自动 激发触发器执行相应的操作。
当对数据表中的数据执行插入、更新和删除操作,需要自动执行一些数据库逻辑时,可以使用触发器来实现。
创建触发器的语法结构是:
CREATE TRIGGER 触发器名称
{BEFORE|AFTER} {INSERT|UPDATE|DELETE} ON 表名
FOR EACH ROW
触发器执行的语句块;
说明:
BEFORE|AFTER
:表示触发的时间。BEFORE
表示在事件之前触发;AFTER
表示在事件之后触发。INSERT | UPDATE|DELETE
:表示触发的事件。
INSERT
表示插入记录时触发;UPDATE
表示更新记录时触发;DELETE
表示删除记录时触发。触发器执行的语句块
:可以是单条SQL语句,也可以是由BEGIN…END结构组成的复合语句块。举例1:
1、创建数据表:
CREATE TABLE test_trigger (
id INT PRIMARY KEY AUTO_INCREMENT,
t_note VARCHAR(30)
);
CREATE TABLE test_trigger_log (
id INT PRIMARY KEY AUTO_INCREMENT,
t_log VARCHAR(30)
);
2、创建触发器:创建名称为before_insert的触发器,向test_trigger数据表插入数据之前,向
test_trigger_log数据表中插入before_insert的日志信息。
DELIMITER //
CREATE TRIGGER before_insert
BEFORE INSERT ON test_trigger
FOR EACH ROW
BEGIN
INSERT INTO test_trigger_log (t_log)
VALUES('before_insert');
END //
DELIMITER ;
3、向test_trigger数据表中插入数据
INSERT INTO test_trigger (t_note) VALUES ('测试 BEFORE INSERT 触发器');
4、查看test_trigger_log数据表中的数据
mysql> SELECT * FROM test_trigger_log;
+----+---------------+
| id | t_log |
+----+---------------+
| 1 | before_insert |
+----+---------------+
1 row in set (0.00 sec)
举例3:
定义触发器“book_check_num”,基于员工表“books”的INSERT事件,在INSERT之前检查将要添加的新书的数量,如果大于当前所有藏书中收藏量最大的书,则报sqlstate_value为’HY000’的错误,从而使得添加失败。
1.建表同时插入数据
CREATE TABLE `book1` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`category_id` tinyint(4) NOT NULL,
`book_name` varchar(255) DEFAULT '',
`num` int(11) DEFAULT '0' COMMENT '书本数量',
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4;
INSERT INTO `mytest`.`book1`(`id`, `category_id`, `book_name`, `num`) VALUES (1, 3, '平凡的世界', 450);
INSERT INTO `mytest`.`book1`(`id`, `category_id`, `book_name`, `num`) VALUES (2, 1, '刺杀小说家', 660);
INSERT INTO `mytest`.`book1`(`id`, `category_id`, `book_name`, `num`) VALUES (3, 2, '鲁滨孙漂流记', 330);
DELIMITER //
CREATE TRIGGER book_check_num
BEFORE INSERT ON book1 FOR EACH ROW
BEGIN
DECLARE max_num INT;
SELECT MAX(num) INTO max_num FROM book1;
IF NEW.num > max_num THEN
SIGNAL SQLSTATE 'HY000' SET MESSAGE_TEXT = '数量大于当前藏书量最大的种类,错误';
END IF;
END //
DELIMITER ;
上面触发器声明过程中的NEW关键字代表INSERT添加语句的新记录。
可以看出表中多了一个触发器
INSERT INTO book1(book_name,num, category_id) VALUES("人生", 1000, 4)
插入失败!
原表中收藏量最大的是 刺杀小说家,660本,所以插入num>660的数据就会报错
INSERT INTO book1(book_name,num, category_id) VALUES("人生", 1000, 4)
> 1644 - 数量大于当前藏书量最大的种类,错误
> 时间: 0.012s
INSERT INTO book1(book_name,num, category_id) VALUES("活着", 200, 4)
插入成功!
INSERT INTO book1(book_name,num, category_id) VALUES("活着", 200, 4)
> Affected rows: 1
> 时间: 0.017s
查看触发器是查看数据库中已经存在的触发器的定义、状态和语法信息等。
方式1:查看当前数据库的所有触发器的定义
SHOW TRIGGERS\G
方式2:查看当前数据库中某个触发器的定义
SHOW CREATE TRIGGER 触发器名
方式3:从系统库information_schema的TRIGGERS表中查询“salary_check_trigger”触发器的信息。
SELECT * FROM information_schema.TRIGGERS;
触发器也是数据库对象,删除触发器也用DROP语句,语法格式如下:
DROP TRIGGER IF EXISTS 触发器名称;
注意,如果在子表中定义了外键约束,并且外键指定了ON UPDATE/DELETE CASCADE/SET NULL子句,此时修改父表被引用的键值或删除父表被引用的记录行时,也会引起子表的修改和删除操作,此时基于子表的UPDATE和DELETE语句定义的触发器并不会被激活。
例如:基于子表员工表(t_employee)的DELETE语句定义了触发器t1,而子表的部门编号(did)字段定
义了外键约束引用了父表部门表(t_department)的主键列部门编号(did),并且该外键加了“ON
DELETE SET NULL”子句,那么如果此时删除父表部门表(t_department)在子表员工表(t_employee)有匹配记录的部门记录时,会引起子表员工表(t_employee)匹配记录的部门编号(did)修改为NULL,但是此时不会激活触发器t1。只有直接对子表员工表(t_employee)执行DELETE语句时才会激活触发器。
意思就是使用了外键,子表添加了on delect触发器,主表数据被删,对应子表字段设置为null,不会启动触发器,只有真的delect数据才会触发。
再次鸣谢康师傅,讲的不错:
https://www.bilibili.com/video/BV1iq4y1u7vj?p=92&spm_id_from=pageDriver&vd_source=a733a89627d9c935525314b7a6581626