MySQL中觸發(fā)器和游標(biāo)的介紹與使用
觸發(fā)器簡(jiǎn)介
觸發(fā)器是和表關(guān)聯(lián)的特殊的存儲(chǔ)過程,可以在插入,刪除或修改表中的數(shù)據(jù)時(shí)觸發(fā)執(zhí)行,比數(shù)據(jù)庫本身標(biāo)準(zhǔn)的功能有更精細(xì)和更復(fù)雜的數(shù)據(jù)控制能力。
觸發(fā)器的優(yōu)點(diǎn):
- 安全性:可以基于數(shù)據(jù)庫的值使用戶具有操作數(shù)據(jù)庫的某種權(quán)利。例如不允許下班后和節(jié)假日修改數(shù)據(jù) 庫數(shù)據(jù);
- 審計(jì):可以跟蹤用戶對(duì)數(shù)據(jù)庫的操作;
- 實(shí)現(xiàn)復(fù)雜的數(shù)據(jù)完整性規(guī)則。例如,觸發(fā)器可回退任何企圖吃進(jìn)超過自己保證金的期貨;
- 提供了運(yùn)行計(jì)劃任務(wù)的另一種方法。例如,如果公司的帳號(hào)上的資金低于 5 萬元?jiǎng)t立即給財(cái)務(wù)人員發(fā)送 警告數(shù)據(jù)。
MySQL 中使用觸發(fā)器
創(chuàng)建觸發(fā)器
創(chuàng)建觸發(fā)器的技巧就是記住觸發(fā)器的四要素:
- 監(jiān)控地點(diǎn):table;
- 監(jiān)控事件:insert/update/delete;
- 觸發(fā)時(shí)間:after/before;
- 觸發(fā)事件:insert/update/delete。
創(chuàng)建觸發(fā)器的基本語法如下所示:
CREATE TRIGGER -- trigger_name:觸發(fā)器的名稱; -- tirgger_time:觸發(fā)時(shí)機(jī),為 BEFORE 或者 AFTER; -- trigger_event:觸發(fā)事件,為 INSERT、DELETE 或者 UPDATE; trigger_name trigger_time trigger_event ON -- tb_name:表示建立觸發(fā)器的表名,在哪張表上建立觸發(fā)器; tb_name -- FOR EACH ROW 表示任何一條記錄上的操作滿足觸發(fā)事件都會(huì)觸發(fā)該觸發(fā)器。 FOR EACH ROW -- trigger_stmt:觸發(fā)器的程序體,可以是一條 SQL 語句或者是用 BEGIN 和 END 包含的多條語句; trigger_stmt
- trigger_name:觸發(fā)器的名稱;
- tirgger_time:觸發(fā)時(shí)機(jī),為 BEFORE 或者 AFTER;
- trigger_event:觸發(fā)事件,為 INSERT、DELETE 或者 UPDATE;
- tb_name:表示建立觸發(fā)器的表名,在哪張表上建立觸發(fā)器;
- trigger_stmt:觸發(fā)器的程序體,可以是一條 SQL 語句或者是用 BEGIN 和 END 包含的多條語句;
- FOR EACH ROW 表示任何一條記錄上的操作滿足觸發(fā)事件都會(huì)觸發(fā)該觸發(fā)器。
注意:對(duì)同一個(gè)表相同觸發(fā)時(shí)間的相同觸發(fā)事件,只能定義一個(gè)觸發(fā)器。
觸發(fā)器新舊記錄
MySQL 中定義了 NEW 和 OLD,用來表示觸發(fā)器的所在表中,觸發(fā)了觸發(fā)器的那一行數(shù)據(jù):
- 在 INSERT 型觸發(fā)器中,NEW 用來表示將要(BEFORE或已經(jīng)(AFTER)插入的新數(shù)據(jù);
- 在 UPDATE型觸發(fā)器中,OLD 用來表示將要或已經(jīng)被修改的原數(shù)據(jù),NEW 用來表示將要或已經(jīng)修改為的新 數(shù)據(jù);
- 在 DELETE型觸發(fā)器中,OLD 用來表示將要或已經(jīng)被刪除的原數(shù)據(jù)。
創(chuàng)建觸發(fā)器,當(dāng)用戶購買商品時(shí),同時(shí)更新對(duì)應(yīng)商品庫存記錄,代碼如下所示:
-- 刪除觸發(fā)器,drop trigger 觸發(fā)器名稱 -- if exists判斷存在才會(huì)刪除 drop trigger if exists myty1; -- 創(chuàng)建觸發(fā)器 create trigger mytg1-- myty1觸發(fā)器的名稱 after insert on orders-- orders在哪張表上建立觸發(fā)器; for each row begin update product set num = num-new.num where pid=new.pid; end; -- 往訂單表插入記錄 insert into orders values(null,2,1); -- 查詢商品表商品庫存更新情況 select * from product;
創(chuàng)建觸發(fā)器,當(dāng)用戶刪除訂單時(shí),同時(shí)更新對(duì)應(yīng)商品庫存記錄,代碼如下所示:
-- 創(chuàng)建觸發(fā)器 create trigger mytg2 after delete on orders for each ROW begin -- 對(duì)庫存進(jìn)行回退,重新加上 update product set num = num+old.num where pid=old.pid; end; -- 刪除訂單記錄 delete from orders where oid = 2; -- 查詢商品表商品庫存更新情況 select * from product;
before 和 after 的區(qū)別
before 在執(zhí)行語句之前after 在執(zhí)行語句之后
當(dāng)訂單商品數(shù)量超過庫存時(shí),修改訂單數(shù)量為最大庫存:
-- -- 創(chuàng)建 before 觸發(fā)器 create trigger mytg3 before insert on orders for each row begin -- 定義一個(gè)變量,來接收庫存 declare n int default 0; -- 查詢庫存 把num賦值給n select num into n from product where pid = new.pid; -- 判斷下單的數(shù)量是否大于庫存量 if new.num>n then -- 大于修改下單庫存(庫存改為最大量) set new.num = n; end if; update product set num = num-new.num where pid=new.pid; end; -- 往訂單表插入記錄 insert into orders values(null,3,50); -- 查詢商品表商品庫存更新情況 select * from product; -- 查詢訂單表 select * from orders;
游標(biāo)
游標(biāo)簡(jiǎn)介
游標(biāo)的作用就是用于對(duì)查詢數(shù)據(jù)庫所返回的記錄進(jìn)行遍歷,以便進(jìn)行相應(yīng)的操作。游標(biāo)有下面這些特征
- 游標(biāo)是只讀的,也就是不能更新它;
- 游標(biāo)是不能滾動(dòng)的,也就是只能在一個(gè)方向上進(jìn)行遍歷,不能在記錄之間隨意進(jìn)退,不能跳過某些記錄;
- 避免在已經(jīng)打開游標(biāo)的表上更新數(shù)據(jù)。
創(chuàng)建游標(biāo)
創(chuàng)建游標(biāo)的語法包含四個(gè)部分:
- 定義游標(biāo):declare 游標(biāo)名 cursor for select 語句;
- 打開游標(biāo):open 游標(biāo)名;
- 獲取結(jié)果:fetch游標(biāo)名 into 變量名[,變量名];
- 關(guān)閉游標(biāo):close 游標(biāo)名;
創(chuàng)建一個(gè)過程 p1,使用游標(biāo)返回 test 數(shù)據(jù)庫中 student 表的第一個(gè)學(xué)生信息。代碼如下所示:
-- 定義過程 create procedure p1() begin declare id int; declare name varchar(20); declare age int; -- 定義游標(biāo) declare 游標(biāo)名 cursor for select 語句; declare mc cursor for select * from student; -- 打開游標(biāo) open 游標(biāo)名; open mc; -- 獲取數(shù)據(jù) fetch 游標(biāo)名 into 變量名[,變量名]; fetch mc into id,name,age; -- 打印 select id,name,age; -- 關(guān)閉游標(biāo) close mc; end; -- 調(diào)用過程 call p1();
在 test 數(shù)據(jù)庫創(chuàng)建一個(gè) student2 表,創(chuàng)建一個(gè)過程 p2,使用游標(biāo)提取 student 表中所有學(xué)生信息插入到 student2 表中。代碼如下所示:
-- 定義過程 create procedure p3() begin declare id int; declare name varchar(20); declare age int; declare flag int default 0; -- 定義游標(biāo) declare 游標(biāo)名 cursor for select 語句; declare mc cursor for select * from student; declare continue handler for not found set flag=1; -- 打開游標(biāo) open 游標(biāo)名; open mc; -- 獲取數(shù)據(jù) fetch 游標(biāo)名 into 變量名[,變量名]; a:loop -- 循環(huán)獲取數(shù)據(jù) fetch mc into id,name,age; if flag=1 then -- 當(dāng)無法fetch時(shí)觸發(fā)continue handler leave a;-- 終止循環(huán) end if; -- 進(jìn)行遍歷,將提取的每一行數(shù)據(jù)插入到 student2 表中 insert into student2 values(id,name,age); end loop; -- 關(guān)閉游標(biāo) close mc; end; -- 調(diào)用過程 call p3(); -- 查詢 student2 表 select * from student2;
總結(jié)
到此這篇關(guān)于MySQL中觸發(fā)器和游標(biāo)的文章就介紹到這了,更多相關(guān)MySQL觸發(fā)器和游標(biāo)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MySQL如何刪除mysql數(shù)據(jù)表內(nèi)的重復(fù)數(shù)據(jù)
這篇文章主要介紹了MySQL如何刪除mysql數(shù)據(jù)表內(nèi)的重復(fù)數(shù)據(jù)問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-04-04
mysql數(shù)據(jù)庫如何導(dǎo)入導(dǎo)出sql文件
這篇文章主要介紹了mysql數(shù)據(jù)庫如何導(dǎo)入導(dǎo)出sql文件問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11
MySQL刪除數(shù)據(jù)后自增主鍵ID不連貫問題及解決
這篇文章主要介紹了MySQL刪除數(shù)據(jù)后自增主鍵ID不連貫問題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-09-09
揭秘SQL優(yōu)化技巧 改善數(shù)據(jù)庫性能
這篇文章是以 MySQL 為背景,很多內(nèi)容同時(shí)適用于其他關(guān)系型數(shù)據(jù)庫,需要有一些索引知識(shí)為基礎(chǔ),重點(diǎn)講述如何優(yōu)化SQL,來提高數(shù)據(jù)庫的性能2012-01-01
關(guān)于mysql?left?join?查詢慢時(shí)間長(zhǎng)的踩坑總結(jié)
這篇文章主要介紹了關(guān)于mysql?left?join?查詢慢時(shí)間長(zhǎng)的踩坑總結(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-09-09
MySQL查看數(shù)據(jù)庫連接數(shù)的方法
本文主要介紹了MySQL查看數(shù)據(jù)庫連接數(shù)的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-08-08
MySQL rand函數(shù)實(shí)現(xiàn)隨機(jī)數(shù)的方法
在mysql中,使用隨機(jī)數(shù)寫一個(gè)語句能一下更新幾百條MYSQL數(shù)據(jù)嗎?答案是肯定的,使用MySQL rand函數(shù),就可以使現(xiàn)在隨機(jī)數(shù)2016-09-09

