所有知識體系文章,GitHub已收錄,歡迎老闆們前來Star! GitHub地址: https://github.com/Ziphtracks/JavaLearningmanual MySQL觸發器 一、什麼是觸發器 觸發器(trigger)是MySQL提供給程式員和數據分析員來保證數據完整性的一種 ...
所有知識體系文章,GitHub已收錄,歡迎老闆們前來Star!
GitHub地址: https://github.com/Ziphtracks/JavaLearningmanual
MySQL觸發器
一、什麼是觸發器
觸發器(trigger)是MySQL提供給程式員和數據分析員來保證數據完整性的一種方法,它是與表事件相關的特殊的存儲過程,它的執行不是由程式調用,也不是手工啟動,而是由事件來觸發,比如當對一個表進行操作(insert,delete, update)時就會激活它執行。簡單理解為:你執行一條sql語句,這條sql語句的執行會自動去觸發執行其他的sql語句。
二、觸發器的作用
- 可在寫入數據表前,強制檢驗或轉換數據。
- 觸發器發生錯誤時,異動的結果會被撤銷。
- 部分資料庫管理系統可以針對數據定義語言(DDL)使用觸發器,稱為DDL觸發器。
- 可依照特定的情況,替換異動的指令 (INSTEAD OF)。
三、觸發器創建的四要素
- 監視地點(table)
- 監視事件(insert、update、delete)
- 觸發時間(after、before)
- 觸發事件(insert、update、delete)
四、觸發器的使用語法
語法:
before/after: 觸發器是在增刪改之前執行,還是之後執行
delete/insert/update: 觸發器由哪些行為觸發(增、刪、改)
on 表名: 觸發器監視哪張表的(增、刪、改)操作
觸發SQL代碼塊: 執行觸發器包含的SQL語句
1CREATE TRIGGER 觸發器名
2BEFORE|AFTER DELETE|INSERT|UPDATE
3ON 表名 FOR EACH ROW
4BEGIN
5觸發SQL代碼塊;
6END;
註意: 觸發器也是存儲過程程式的一種,而觸發器內部的執行SQL語句是可以多行操作的,所以在MySQL的存儲過程程式中,要定義結束符。
如果MySQL存儲過程不瞭解的小伙伴,可以參考此文面向MySQL存儲過程編程,文章中詳細講解了MySQL存儲過程的優勢和語法等等,相信你會在其中得以收穫。
1# 設置MySQL執行結束標誌,預設為;
2delimiter //
五、觸發器的基本使用
5.1 基本使用步驟
首先,我先展示一下創建的兩張表,因為創建的表很簡單,這裡我沒有提供庫表操作的SQL命令。
tb_class
employee
其次,創建了一個含有update操作的存儲過程
1delimiter //
2create procedure update_emp(in i int, in p int)
3begin
4 update employee set phone = p where id = i;
5end //
再創建一個觸發器
分析: 觸發器名稱為t1,觸發時間為after,監視動作為update,監視表為employee表。彙總一起解釋這個觸發器就是:創建一個觸發器名稱為t1的觸發器,觸發器監視employee表執行update(更新)操作後,就開始執行觸發器內部SQL語句
update tb_class set num = num + 1 where id = 1;
。簡單來說就是一個監視一個表的增、刪、改操作並設置操作前後時間,在設置時間的範圍內對另外一個表進行其他操作。
如果你學到這裡還是一知半解,後面我會講解一個訂單與庫存的數據關係,到那時候你就會明白了!
1delimiter //
2# 創建觸發器,觸發器名稱為t1
3create trigger t1
4 # 觸發器執行在update操作之後
5 after update
6 # 監視employee表
7 on employee
8 for each row
9begin
10 # 觸發執行的SQL語句
11 update tb_class set num = num + 1 where id = 1;
12end //
最後調用函數,並查看、分析結果
1call update_emp(2, 110);
觸發器在此場景的作用分析
當employee表發生update操作時,觸發器就對tb_class表中的num值做修改。
執行結果發現,我們在使用函數將employee表中id為2員工的phone修改為110後,觸發器監視到employee表中發生了update更新操作,就執行了內部SQL語句,也就是將tb_class表中id為1的num值自增1。
5.2 查看和刪除已有的觸發器
查看已有觸發器:
show triggers
刪除已有觸發器:
drop trigger 觸發器名稱
5.3 for each row
這裡擴展,在oracle觸發器中,觸發器分為行觸發器和語句觸發器。也就是說,假設你監視一個修改操作,它修改了1000行代碼,在Oracle中觸發器會觸發1000次。
在oracle中,for each row如果不寫,無論update語句一次影響了多少行,都只執行一次觸發事件。
而MySQL中,不支持語句級觸發器,所以在MySQL中並不需要在意。
六、訂單與庫存關係場景
訂單與庫存的關係: 用戶下訂單,意味著創建該商品訂單,該商品訂單中的商品數量為1,庫存中的該商品數量-1。往往訂單表和庫存表中的數量是同時操作的,所以我們這裡可以用觸發器。
觸發器應用: 關於訂單表,下訂單肯定是涉及到insert插入數據數量的操作。我們可以創建一個監視訂單表insert操作後執行庫存表數量-1的觸發器來完成訂單與庫存表的同時修改。
創建表,併在表中添加幾條數據:
1create table goods(
2 gid int,
3 name varchar(20),
4 num smallint
5);
6create table ord(
7 oid int,
8 gid int,
9 much smallint
10);
11insert into goods values(1,'cat',40);
12insert into goods values(2,'dog',63);
13insert into goods values(3,'pig',87);
創建觸發器
1create trigger t1
2after
3insert
4on ord
5for each row
6begin
7 update goods set num = num - 1 where gid = 1;
8end$
該觸發器意為,用戶不管下什麼訂單,都會把商品編號為1的商品的庫存減去1。
七、觸發器中引用行變數
7.1 old和new對象語法
- 在觸發目標上執行insert操作後會有一個新行,如果在觸發事件中需要用到這個新行的變數,可以用new關鍵字表示
- 在觸發目標上執行delete操作後會有一個舊行,如果在觸發事件中需要用到這個舊行的變數,可以用old關鍵字表示
- 在觸發目標上執行update操作後原紀錄是舊行,新記錄是新行,可以使用new和old關鍵字來分別操作