資料庫 觸發器

2022-07-13 06:36:11 字數 4929 閱讀 1828

觸發器時為執行業務規則和保持資料完整性而提供的一種機制,它可以在執行插入、更新,刪除等操作的前後自動觸發。觸發器與儲存過程類似,但是讓不接收輸入\輸出引數沒也不能被顯式呼叫,只能有伺服器事件自動觸發,根據一起執行觸發器語言的不同,可將其分為dml觸發器和ddl觸發器

1、dml觸發器

根據dml觸發器發生的時間,編寫觸發器所使用的語言,可以分為after觸發器、instead of觸發器和clr觸發器。alter觸發器在執行insert、update或者delete語句操作之後,instead of 觸發器和約束之後觸發,instead of在處理約束前激發,因此可以再instead of 中使用其他語句來替代激發觸發器的insert、update等語句,並且,還可以為基於乙個或者多個基表的檢視定義instead of 觸發器,從而擴充套件檢視可支援的更新型別,clr觸發器可以是after觸發器或者是instead of 觸發器,也可以是ddl觸發器,需要注意的是,在建立dml觸發器時,不能使用一下語句:

alter database; create database;drop database;

load database load log; reconfigure

restore database restore log

(1)    after觸發器

乙個表中可以有多個after觸發器,只要他們的名稱不同即可。每個觸發器只能應用於乙個表,但是乙個觸發器可以應用於乙個表的三個使用者操作(update、insert、delete)

下面的雨季建立了乙個pritrigger,和乙個detailtable,其中pritrigger表用於存放銷售訂單的編號和金額,detailtable表用於存放每筆訂單的產品資訊。為pritrigger表中的delete操作建立了乙個名為pritrigger的觸發器,當刪除pritrigger表中的訂單資訊時,該觸發器將刪除detailtable表中該訂單的產品資訊

--建立主表,存放訂單編號和金額

create table pritrigger

( orderid int identity(1,1),ordertotal money)

--明細表 存放訂單中的產品資訊

create table detailtable

( orderid int,

productid int,

productcount int,

price money)

insert into pritrigger values(100)

insert into pritrigger values(200)

insert into detailtable values(1,1,10,110)

insert into detailtable values(1,2,10,1000)

insert into detailtable values(2,2,10,1000)

create trigger pritrigger1

on pritrigger

after delete

asdelete from detailtable

where orderid in(

select orderid from deleted

在定義觸發器時,觸發器名稱在create trigger關鍵字之後,on字句指定要建立觸發器的基表,after字句(也可以使用for來代替after關鍵字,二者功能相同)指定啟用觸發器的操作語句,可以同時指定多個操作語句。例如「after delete,insert」標示在對錶執行delete、insert 語句時啟用觸發器。as關鍵字指定觸發器執行什麼樣的操作、注意where條件中in字句中的deleted關鍵字。當從pritrigger表中刪除行時,被刪除的行會複製到乙個名為deleted的臨時記憶體表中,如果為表指定了乙個insert語句時的觸發器,則在想表中插入行時,新行將同時被新增到乙個名為inserted的臨時記憶體表中。如果為表指定了乙個執行update語句時的觸發器。由於更新事務類似於在刪除操作之後執行插入操作。因此,舊的資料行被複製到deleted表中,然後新行複製到觸發器表和inserted表中

deleted表和inserted表都是由資料庫引擎自動建立和管理的,這些表的結構與定義觸發器的基表的結構相同

(2)    instead of 觸發器

sqlserver伺服器在執行after觸發器的sql**後,先建立臨時的inserted表和deleted表,然後執行**中對資料庫操作,最後才啟用觸發器中的**。而對於替代(instead of)觸發器,sqlserver伺服器在執行觸發instead of 觸發器的**時,先建立臨時的inserted表和deleted表,然後直接觸發instead of觸發器,而拒絕執行使用者輸入的dml操作語句。

--建立instead of 觸發器

create trigger trig_insteadof

on student

instead of insert

asbegin

declare @stuage int;

select @stuage=(select stu_age from inserted)

if(@stuage >120)

select '插入年齡錯誤' as '失敗原因'

end(3)    巢狀觸發器

如果乙個觸發器在執行操作時呼叫了另外乙個觸發器,而這個觸發器又接著呼叫了下乙個觸發器,那麼就形成了巢狀觸發器。巢狀觸發器在安裝時就被啟用,但是可以使用系統儲存過程sp_configure禁用和重新啟用巢狀觸發器。

巢狀觸發器不一定要形成乙個環,它可以 t1->t2->t3...這樣一直觸發下去,最多允許巢狀 32 層。如果巢狀的次數超過限制,那麼該觸發器將被終止,並回滾整個事務,使用巢狀觸發器需要注意以下幾點:

預設情況下,巢狀觸發器配置選項是開啟的。

在同乙個觸發器事務中,乙個巢狀觸發器不能被觸發兩次。

由於觸發器是乙個事務,如果在一系列巢狀觸發器的任意層次中發生錯誤,則整個事物都將取消,而且所有資料回滾。

巢狀是用來保持整個資料庫的完整性的重要功能,但有時可能需要禁用巢狀,如果禁用了巢狀,那麼修改乙個觸發器的實現不會再觸發該錶上的任何觸發器。在下述情況下,需要禁用巢狀觸發器:

巢狀觸發要求複雜而有理論的設計,級聯修改可能會修改使用者不想涉及的資料。

在一系列巢狀觸發器中的任意點的時間修改操作都會觸發一些觸發器,儘管這時資料庫提供很強的保護功能,但如果以特定的順序更新表,就會產生問題。

(4)遞迴觸發器

觸發器的遞迴是指乙個觸發器從其內部再一次啟用該觸發器,例如update操作啟用的觸發器內部還有一條資料表的更新語句,那麼這個更新語句就有可能啟用這個觸發器本身,當然,這種遞迴的觸發器內部還會有判斷語句,只有一定情況下才會執行那個t_sql語句,否則就成為無線呼叫的死迴圈了。

sqlserver中的遞迴觸發器包括兩種:直接遞迴和間接遞迴。

直接遞迴:觸發器被觸發後並執行乙個操作,而該操作又使用乙個觸發器再次被觸發。

間接遞迴:觸發器被觸發並執行乙個操作,而該操作又使另乙個表中的某個觸發器被觸發,第二個觸發器使原始表得到更新,從而再次觸發第乙個觸發器。

預設情況下,遞迴觸發器選項是禁用的。遞迴觸發器最多只能遞迴16層,如果遞迴中的第16個觸發器啟用了第17個觸發器,則結果與發布的rollback命令一樣,所有資料都將回滾。 

我們舉例解釋如下,假如有表1、表2名稱分別為 t1、t2,在 t1、t2 上分別有觸發器 g1、g2。

間接遞迴:對 t1 操作從而觸發 g1,g1 對 t2 操作從而觸發 g2,g2 對 t1 操作從而再次觸發 g1...

直接遞迴:對 t1 操作從而觸發 g1,g1 對 t1 操作從而再次觸發 g1... 

設定直接遞迴:

預設情況下是禁止直接遞迴的,要設定為允許有兩種方法:

t-sql:exec sp_dboption 'dbname', 'recursive triggers', true;

em:資料庫上點右鍵->屬性->選項。 

2、管理觸發器

1.檢視觸發器

(1).檢視資料庫中所有的觸發器

--檢視資料庫中所有的觸發器

use 資料庫名

goselect * from sysobjects where xtype='tr'

sysobjects 儲存著資料庫的物件,其中 xtype 為 tr 的記錄即為觸發器物件。在 name 一列,我們可以看到觸發器名稱。

(2).sp_helptext 檢視觸發器內容

use 資料庫名

goexec sp_helptext '觸發器名稱'

將會以表的樣式顯示觸發器內容。 

除了觸發器外,sp_helptext 還可以顯示 規則、預設值、未加密的儲存過程、使用者定義函式、檢視的文字。

(3).sp_helptrigger 用於檢視觸發器的屬性

sp_helptrigger 有兩個引數:第乙個引數為表名;第二個為觸發器型別,為 char(6) 型別,可以是 insert、update、delete,如果省略則顯示指定表中所有型別觸發器的屬性。

use 資料庫名

goexec sp_helptrigger tablename

2.禁用啟用觸發器

禁用:alter table 表名 disable trigger 觸發器名稱

啟用:alter table 表名 enable trigger 觸發器名稱

如果有多個觸發器,則各個觸發器名稱之間用英文逗號隔開。

如果把「觸發器名稱」換成「all」,則表示禁用或啟用該錶的全部觸發器。

3修改觸發器

--修改觸發器語法

alter trigger  trigger_name

on  table_name

[ with encryption ]

for

assql_statement;

4.刪除觸發器

--語法格式:

drop  trigger   [ ,...n ]

引數:trigger: 要刪除的觸發器名稱

n:表示可以刪除多個觸發器的佔位符      

資料庫觸發器

觸發器是一種特殊型別的儲存過程,它不同於我們前面介紹過的儲存過程。觸發器主要是通過事件進行觸發而被執行的,而儲存過程可以通過儲存過程名字而 被直接呼叫。當對某一表進行諸如update insert delete 這些操作時,sql server 就會自動執行觸發器所定義的sql 語句,從而確保對資料...

資料庫觸發器

最近做了個觸發器的例子 create trigger tru user on user for update asif update status begin update user set stopflag 1 from inserted where user.userid inserted.us...

資料庫 觸發器

觸發器的概念 是使用者定義在關係表上的一類有事件驅動的特殊過程。一旦定義,任何對錶的增刪改操作均有伺服器自動啟用相應的觸發器,在dbms核心層進行集中的完整性控制。類似於約束,但比約束更靈活。觸發器的分類 dml觸發器 dml data manipulation language 觸發器是當資料庫伺...