MySQL触法器
1. 什么是触发器及其特点
MySQL数据库中触发器是一个特殊的存储过程,不同的是执行存储过程要使用 CALL 语句来调用,而触发器的执行不需要使用 CALL 语句来调用,也不需要手工启动,只要一个预定义的事件发生就会被 MySQL自动调用。
2. 触发器优缺点
优点:
- 触发程序的执行是自动的,当对触发程序相关表的数据做出相应的修改后立即执行
- 触发程序可以通过数据库中相关的表层叠修改另外的表
- 触发程序可以实施比 FOREIGN KEY 约束、CHECK 约束更为复杂的检查和操作
缺点:
- 随着数据库中触发器数量的增加,系统的复杂性会显著提高
- 如果触发器编写不当,可能会导致死锁等问题
- 触发器中的业务逻辑是隐藏在数据库中,逻辑不容易被开发人员和维护人员直观地看到
- 特别是当触发器中包含嵌套查询、大量的数据更新操作等复杂情况时,可能会导致数据库的性能瓶颈。而且,由于触发器是自动执行的,有时候很难预测它们在高并发环境下对系统性能的具体影响程度。
3. 引发触发器的事件
- 在 INSERT 语句执行之前或之后响应的触发器
- 在 UPDATE 语句执行之前或之后响应的触发器
- 在 DELETE 语句执行之前或之后响应的触发器
4. 触发器语法
CREATE TRIGGER <触发器名> < BEFORE | AFTER >
<INSERT | UPDATE | DELETE >
ON <表名> FOR EACH Row<触发器主体>
FOR EACH Row:一般是指行级触发,对于受触发事件影响的每一行都要激活触发器的动作。例如,使用 INSERT 语句向某个表中插入多行数据时,触发器会对每一行数据的插入都执行相应的触发器动作。
5. 触发器实例
基于Mysql5.7编写一个触发器,实现当增加或修改订单明细中的金额时自动修改订单的总金额。
- 创建表
CREATE TABLE orders (
id INT AUTO_INCREMENT PRIMARY KEY,
total_amount DECIMAL(10, 2) DEFAULT 0.00
);
CREATE TABLE order_details (
id INT AUTO_INCREMENT PRIMARY KEY,
order_id INT,
amount DECIMAL(10, 2) NOT NULL,
FOREIGN KEY (order_id) REFERENCES orders(id) ON DELETE CASCADE
);
- 创建insert触发器
DELIMITER //
CREATE TRIGGER after_order_detail_insert
AFTER INSERT ON order_details
FOR EACH ROW
BEGIN
UPDATE orders
SET total_amount = total_amount + NEW.amount
WHERE id = NEW.order_id;
END;
//
DELIMITER ;
使用 INSERT 触发器需要注意以下几点:
- 在 INSERT 触发器代码内,可引用一个名为 NEW(不区分大小写)的虚拟表来访问被插入的行。
- 在 BEFORE INSERT 触发器中,NEW 中的值也可以被更新,即允许更改被插入的值(只要具有对应的操作权限)。
- 对于 AUTO_INCREMENT 列,NEW 在 INSERT 执行之前包含的值是 0,在 INSERT 执行之后将包含新的自动生成值。
- 创建update触发器
DELIMITER //
CREATE TRIGGER after_order_detail_update
AFTER UPDATE ON order_details
FOR EACH ROW
BEGIN
UPDATE orders
SET total_amount = total_amount - OLD.amount + NEW.amount
WHERE id = NEW.order_id;
END;
//
DELIMITER ;
使用 UPDATE 触发器需要注意以下几点:
- 在 UPDATE 触发器代码内,可引用一个名为 NEW(不区分大小写)的虚拟表来访问更新的值。
- 在 UPDATE 触发器代码内,可引用一个名为 OLD(不区分大小写)的虚拟表来访问 UPDATE 语句执行前的值。
- 在 BEFORE UPDATE 触发器中,NEW 中的值可能也被更新,即允许更改将要用于 UPDATE 语句中的值(只要具有对应的操作权限)。
- OLD 中的值全部是只读的,不能被更新。
- 删除触发器
DELIMITER //
CREATE TRIGGER after_order_detail_delete
AFTER DELETE ON order_details
FOR EACH ROW
BEGIN
UPDATE orders
SET total_amount = total_amount - OLD.amount
WHERE id = OLD.order_id;
END;
//
DELIMITER ;
使用 DELETE 触发器需要注意以下几点:
- 在 DELETE 触发器代码内,可以引用一个名为 OLD(不区分大小写)的虚拟表来访问被删除的行。
- OLD 中的值全部是只读的,不能被更新。
- 插入及更新测试
-- 插入订单
INSERT INTO orders (total_amount) VALUES (0);
-- 插入明细
INSERT INTO order_details (order_id, amount) VALUES (1, 100.00);
INSERT INTO order_details (order_id, amount) VALUES (1, 50.00);
-- 更新明细
UPDATE order_details SET amount = 120.00 WHERE id = 1;
DELETE FROM order_details WHERE id = 2;
SELECT * FROM orders;