在SQL中,名詞觸發器指“在數據庫中為響應一個特殊表格中的某些事件而自動執行的程序代碼。”(Wikipedia)說得簡單一些,它是在一個特殊的數據庫事件,如INSERT或DELETE發生時,自動激活的一段代碼。觸發器可方便地用於日志記錄、對單個表格到其他鏈接式表格進行自動的“層疊式”更改、或保證對表格關系進行自動更新。當一個新整數值增加到數據庫域中時,自動更新運行的總數的代碼段是一個觸發器。自動記錄對一個特殊數據庫表格所作更改的SQL命令塊也是一個觸發器實例。
觸發器是MySQL 5.x的新功能,隨著5.x代碼樹新版本的出現,這一功能也逐漸得到改善。在本文中,我將簡單介紹如何定義並使用觸發器,查看觸發器狀態,並如何在使用完畢後刪除觸發器。我還將為你展示一個觸發器在現實世界中的應用實例,並檢驗它對數據庫記錄的改變。
一個簡單實例
通過簡單(雖然是人為的)實例來說明是了解MySQL觸發器應用的最佳辦法。首先我們建立兩個單域的表格。一個表格中為姓名列表(表格名:data),另一個表格中是所插入字符的字符數(表格名:chars)。我希望在data表格中定義一個觸發器,每次在其中插入一個新姓名時,chars表格中運行的總數就會根據新插入記錄的字符數目進行自動更新。
MySQL> CREATE TABLE data (name VARCHAR(255));
Query OK, 0 rows affected (0.09 sec)
MySQL> CREATE TABLE chars (count INT(10));
Query OK, 0 rows affected (0.07 sec)
MySQL> INSERT INTO chars (count) VALUES (0);
Query OK, 1 row affected (0.00 sec)
MySQL> CREATE TRIGGER t1 AFTER INSERT ON
data FOR EACH ROW UPDATE chars SET count = count + CHAR_LENGTH(NEW.name);
Query OK, 0 rows affected (0.01 sec)
理解上面代碼的關鍵在於CREATE TRIGGER命令,它被用來定義一個新觸發器。這個命令建立一個新觸發器,假定的名稱為t1,每次有一個新記錄插入到data表格中時,t1就被激活。
在這個觸發器中有兩個重要的子句:
AFTER INSERT子句表明觸發器在新記錄插入data表格後激活。
UPDATE chars SET count = count + CHAR_LENGTH(NEW.name)子句表示觸發器激活後執行的SQL命令。在本例中,該命令表明用新插入的data.name域的字符數來更新 chars.count欄。這一信息可通過內置的MySQL函數CHAR_LENGTH()獲得。
放在源表格域名前面的NEW關鍵字也值得注意。這個關鍵字表明觸發器應考慮域的new值(也就是說,剛被插入到域中的值)。MySQL還支持相應的OLD前綴,可用它來指域以前的值。
你可以通過調用SHOW TRIGGER命令來檢查觸發器是否被激活。
MySQL> SHOW TRIGGERS\G
*************************** 1. row ***************************
?Trigger: t1
?Event: INSERT
?Table: data
Statement: UPDATE chars SET count = count + CHAR_LENGTH(NEW.name)
Timing: AFTER
?Created: NULL
ql_mode:
1 row in set (0.01 sec)
MySQL> CREATE TRIGGER t1 AFTER UPDATEON accounts
FOR EACH ROW INSERT INTO audit (id, balance, user, time)
VALUES (OLD.id, NEW.balance, CURRENT_USER(), NOW());
Query OK, 0 rows affected (0.04 sec)
如果你已經走到這一步,就很容易理解。accounts表格每經歷一次UPDATE,觸發器插入(INSERT)對應記錄的id、新的余額、當前時間和登錄audit表格的用戶的名稱。
實現中的例子用觸發器審計記錄,既然你了解了觸發器的基本原理,讓我們來看一個稍稍復雜的例子。我們常用觸發器來建立一個自動“審計記錄”,以記錄各種用戶對數據庫的更改。為了解審計記錄的實際應用,請看下面的表格(表格名:accounts),它列出了一個用戶的三個銀行賬戶余額。
MySQL> SELECT * FROM accounts;
+----+------------+---------+
| id | label| balance |
+----+------------+---------+
|1 | Savings #1 |500 |
|2 | Current #1 |2000 |
|3 | Current #2 |3500 |
+----+------------+---------+
3 rows in set (0.00 sec)
然後,檢查觸發器是否被激活:
MySQL> SHOW TRIGGERS \G
*************************** 1. row ***************************
?Trigger: t1
?Event: UPDATE
?Table: accounts
Statement: INSERT INTO audit (id, balance, user, time)
VALUES (OLD.id, NEW.balance, CURRENT_USER(), NOW())
Timing: AFTER
?Created: NULL
Sql_mode:
1 row in set (0.01 sec)
再來看最後的結果:
MySQL> UPDATE accounts SET balance = 500 WHERE id = 1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1?Changed: 1?Warnings: 0
MySQL> UPDATE accounts SET balance = 900 WHERE id = 3;
Query OK, 1 row affected (0.01 sec)
Rows matched: 1?Changed: 1?Warnings: 0
MySQL> UPDATE accounts SET balance = 1900 WHERE id = 1;
Query OK, 1 row affected (0.00 sec)
Rows matched: 1?Changed: 1?Warnings: 0
注意,對accounts表格所作的改變已被記錄到audit表格中,將來如果出現問題,我們可以方便地從中進行恢復。
MySQL> SELECT * FROM audit;
+------+---------+----------------+---------------------+
| id| balance | user| time|
+------+---------+----------------+---------------------+
|1 |500 | root@localhost | 2006-04-22 12:52:15 |
|3 |900 | root@localhost | 2006-04-22 12:53:15 |
|1 |1900 | root@localhost | 2006-04-22 12:53:23 |
+------+---------+----------------+---------------------+
3 rows in set (0.00 sec)
如上面的例子所示,觸發器是一個強大的新功能,它大大增強了RDBMS的自動化程度。