MySQL好像從5.0.2版本就開始支持觸發器的功能了,本次博客就來介紹一下觸發器,首先仍是談下概念性的東西吧:php
什麼是觸發器
觸發器是與表有關的數據庫對象,在知足定義條件時觸發,並執行觸發器中定義的語句集合。觸發器的這種特性能夠協助應用在數據庫端確保數據的完整性。html
舉個例子,好比你如今有兩個表【用戶表】和【日誌表】,當一個用戶被建立的時候,就須要在日誌表中插入建立的log日誌,若是在不使用觸發器的狀況下,你須要編寫程序語言邏輯才能實現,可是若是你定義了一個觸發器,觸發器的做用就是當你在用戶表中插入一條數據的以後幫你在日誌表中插入一條日誌信息。固然觸發器並非只能進行插入操做,還能執行修改,刪除。mysql
建立觸發器
建立觸發器的語法以下:sql
CREATE TRIGGER trigger_name trigger_time trigger_event ON tb_name FOR EACH ROW trigger_stmt trigger_name:觸發器的名稱 tirgger_time:觸發時機,爲BEFORE或者AFTER trigger_event:觸發事件,爲INSERT、DELETE或者UPDATE tb_name:表示創建觸發器的代表,就是在哪張表上創建觸發器 trigger_stmt:觸發器的程序體,能夠是一條SQL語句或者是用BEGIN和END包含的多條語句 因此能夠說MySQL建立如下六種觸發器: BEFORE INSERT,BEFORE DELETE,BEFORE UPDATE AFTER INSERT,AFTER DELETE,AFTER UPDATE
其中,觸發器名參數指要建立的觸發器的名字數據庫
BEFORE和AFTER參數指定了觸發執行的時間,在事件以前或是以後安全
FOR EACH ROW表示任何一條記錄上的操做知足觸發事件都會觸發該觸發器函數
建立有多個執行語句的觸發器
CREATE TRIGGER 觸發器名 BEFORE|AFTER 觸發事件 ON 表名 FOR EACH ROW BEGIN 執行語句列表 END
其中,BEGIN與END之間的執行語句列表參數表示須要執行的多個語句,不一樣語句用分號隔開post
tips:通常狀況下,mysql默認是以 ; 做爲結束執行語句,與觸發器中須要的分行起衝突測試
爲解決此問題可用DELIMITER,如:DELIMITER ||,能夠將結束符號變成||編碼
當觸發器建立完成後,能夠用DELIMITER ;來將結束符號變成;
mysql> DELIMITER || mysql> CREATE TRIGGER demo BEFORE DELETE -> ON users FOR EACH ROW -> BEGIN -> INSERT INTO logs VALUES(NOW()); -> INSERT INTO logs VALUES(NOW()); -> END -> || Query OK, 0 rows affected (0.06 sec) mysql> DELIMITER ;
上面的語句中,開頭將結束符號定義爲||,中間定義一個觸發器,一旦有知足條件的刪除操做
就會執行BEGIN和END中的語句,接着使用||結束
最後使用DELIMITER ; 將結束符號還原
tigger_event:
load data語句是將文件的內容插入到表中,至關因而insert語句,而replace語句在通常的狀況下和insert差很少,可是若是表中存在primary 或者unique索引的時候,若是插入的數據和原來的primary key或者unique相同的時候,會刪除原來的數據,而後增長一條新的數據,因此有的時候執行一條replace語句至關於執行了一條delete和insert語句。
觸發器能夠是一條SQL語句,也能夠是多條SQL代碼塊,那如何建立呢?
DELIMITER $ #將語句的分隔符改成$ BEGIN sql1; sql2; ... sqln END $ DELIMITER ; #將語句的分隔符改回原來的分號";"
在BEGIN...END語句中也能夠定義變量,可是隻能在BEGIN...END內部使用:
DECLARE var_name var_type [DEFAULT value] #定義變量,可指定默認值 SET var_name = value #給變量賦值
NEW和OLD的使用:
根據以上的表格,可使用一下格式來使用相應的數據:
NEW.columnname:新增行的某列數據 OLD.columnname:刪除行的某列數據
說了這麼多如今咱們來建立一個觸發器吧!
如今有表以下:
用戶users表
CREATE TABLE `users` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(255) CHARACTER SET utf8mb4 DEFAULT NULL, `add_time` int(11) DEFAULT NULL, PRIMARY KEY (`id`), KEY `name` (`name`(250)) USING BTREE ) ENGINE=MyISAM AUTO_INCREMENT=1000001 DEFAULT CHARSET=latin1;
日誌logs表:
CREATE TABLE `logs` ( `Id` int(11) NOT NULL AUTO_INCREMENT, `log` varchar(255) DEFAULT NULL COMMENT '日誌說明', PRIMARY KEY (`Id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='日誌表';
需求是:當在users中插入一條數據,就會在logs中生成一條日誌信息。
建立觸發器:
DELIMITER $ CREATE TRIGGER user_log AFTER INSERT ON users FOR EACH ROW BEGIN DECLARE s1 VARCHAR(40)character set utf8; DECLARE s2 VARCHAR(20) character set utf8;#後面發現中文字符編碼出現亂碼,這裏設置字符集 SET s2 = " is created"; SET s1 = CONCAT(NEW.name,s2); #函數CONCAT能夠將字符串鏈接 INSERT INTO logs(log) values(s1); END $ DELIMITER ;
這裏我用的navicat:
查看觸發器
SHOW TRIGGERS語句查看觸發器信息
Tip:
上面我用的navicat直接建立,若是你們用的mysql front,name這裏會有個區別,咱們刪除剛纔的觸發器,在Mysql front中測試
drop trigger user_log;#刪除觸發器
打開Mysql Front:
mysql front在編譯sql時,不用定義結尾分隔符,修改後的sql直接這樣既可:
#DELIMITER $ CREATE TRIGGER user_log AFTER INSERT ON users FOR EACH ROW BEGIN DECLARE s1 VARCHAR(40)character set utf8; DECLARE s2 VARCHAR(20) character set utf8; SET s2 = " is created"; SET s1 = CONCAT(NEW.name,s2); #函數CONCAT能夠將字符串鏈接 INSERT INTO logs(log) values(s1); END #$ #DELIMITER ;
這裏再囉嗦幾句:
tips:SHOW TRIGGERS語句沒法查詢指定的觸發器
在triggers表中查看觸發器信息
SELECT * FROM information_schema.triggers;
結果顯示了全部觸發器的詳細信息,同時,該方法能夠查詢制定觸發器的詳細信息
SELECT * FROM information_schema.triggers WHERE TRIGGER_NAME='user_log';
tips:全部觸發器信息都存儲在information_schema數據庫下的triggers表中
可使用SELECT語句查詢,若是觸發器信息過多,最好經過TRIGGER_NAME字段指定查詢
回到上面,咱們建立好了觸發器,繼續在users中插入數據並查看數據:
insert into users(name,add_time) values('周伯通',now());
好吧,咱們再來查看一下logs表吧!
經過上面的例子,能夠看到只須要在users中插入用戶的信息,日誌會自動記錄到logs表中,這也許就是觸發器給我帶來的便捷吧!
限制和注意事項
觸發器會有如下兩種限制:
1.觸發程序不能調用將數據返回客戶端的存儲程序,也不能使用採用CALL語句的動態SQL語句,可是容許存儲程序經過參數將數據返回觸發程序,也就是存儲過程或者函數經過OUT或者INOUT類型的參數將數據返回觸發器是能夠的,可是不能調用直接返回數據的過程。
2.不能再觸發器中使用以顯示或隱式方式開始或結束事務的語句,如START TRANS-ACTION,COMMIT或ROLLBACK。
注意事項:MySQL的觸發器是按照BEFORE觸發器、行操做、AFTER觸發器的順序執行的,其中任何一步發生錯誤都不會繼續執行剩下的操做,若是對事務表進行的操做,若是出現錯誤,那麼將會被回滾,若是是對非事務表進行操做,那麼就沒法回滾了,數據可能會出錯。
總結
觸發器是基於行觸發的,因此刪除、新增或者修改操做可能都會激活觸發器,因此不要編寫過於複雜的觸發器,也不要增長過得的觸發器,這樣會對數據的插入、修改或者刪除帶來比較嚴重的影響,同時也會帶來可移植性差的後果,因此在設計觸發器的時候必定要有所考慮。
觸發器是一種特殊的存儲過程,它在插入,刪除或修改特定表中的數據時觸發執行,它比數據庫自己標準的功能有更精細和更復雜的數據控制能力。
數據庫觸發器有如下的做用:
1.安全性。能夠基於數據庫的值使用戶具備操做數據庫的某種權利。
# 能夠基於時間限制用戶的操做,例如不容許下班後和節假日修改數據庫數據。
# 能夠基於數據庫中的數據限制用戶的操做,例如不容許股票的價格的升幅一次超過10%。
2.審計。能夠跟蹤用戶對數據庫的操做。
# 審計用戶操做數據庫的語句。
# 把用戶對數據庫的更新寫入審計表。
3.實現複雜的數據完整性規則
# 實現非標準的數據完整性檢查和約束。觸發器可產生比規則更爲複雜的限制。與規則不一樣,觸發器能夠引用列或數據庫對象。例如,觸發器可回退任何企圖吃進超過本身保證金的期貨。
# 提供可變的缺省值。
4.實現複雜的非標準的數據庫相關完整性規則。觸發器能夠對數據庫中相關的表進行連環更新。例如,在auths表author_code列上的刪除觸發器可致使相應刪除在其它表中的與之匹配的行。
# 在修改或刪除時級聯修改或刪除其它表中的與之匹配的行。
# 在修改或刪除時把其它表中的與之匹配的行設成NULL值。
# 在修改或刪除時把其它表中的與之匹配的行級聯設成缺省值。
# 觸發器可以拒絕或回退那些破壞相關完整性的變化,取消試圖進行數據更新的事務。當插入一個與其主健不匹配的外部鍵時,這種觸發器會起做用。例如,能夠在books.author_code 列上生成一個插入觸發器,若是新值與auths.author_code列中的某值不匹配時,插入被回退。
5.同步實時地複製表中的數據。
6.自動計算數據值,若是數據的值達到了必定的要求,則進行特定的處理。例如,若是公司的賬號上的資金低於5萬元則當即給財務人員發送警告數據。