前觸發(fā)器和后觸發(fā)器簡介(downmoon)
??? 觸發(fā)器是一種特殊的存儲過程。當(dāng)Insert Update 或者Delete 語句修改表中一個(gè)或者多個(gè)行時(shí)執(zhí)行觸發(fā)器。因?yàn)镾QL Server 對特定表上的每一個(gè)指定操作調(diào)用一個(gè)觸發(fā)器,所以可以使用觸發(fā)器擴(kuò)展SQL Sever 的內(nèi)置完整性和數(shù)據(jù)操縱功能.
??? 注意: 不像Delete 語句,Trancate Table 語句不激活觸發(fā)器,Write Text 語句也不激活觸發(fā)器。
??? 在SQL Sever 2000 中支持兩種類型的觸發(fā)器,前觸發(fā)器(Instead Of Trigger)和后觸發(fā)器(After Trigger)。前觸發(fā)器就是在語句執(zhí)行之前激活觸發(fā)器,而后觸發(fā)器就是在語句執(zhí)行之后激活觸發(fā)器。可以通過FOR 子句來選擇使用何種觸發(fā)器。
??? 當(dāng)為每一種操作創(chuàng)建一個(gè)觸發(fā)器時(shí),可以為所有三種操作創(chuàng)建一個(gè)觸發(fā)器,并且使用相應(yīng)的編程技術(shù)處理每一種操作。下面的示例在For 子句中列出了三種語句類型并,且使用條件語句將相應(yīng)的跟蹤值插入到CustUpdLog 表中。
Create Trigger TrackCustomerUpdates
???On AppDta.dbo.Customer
??????For Insert,Update,Delete
???As
??????Declare @InsertedCount Int
??????Declare @DeletedCount Int
??????Set @InsertedCount=(Select Count(*)From inserted)
??????Set @DeletedCount=(Select Count(*)From deleted)
??????If ( @InsertedCount>0)
?????????Begin
????????????Insert Into AppDta.dbo.CustUpdLog
????????????( CustID,
???????????????Action,
???????????????UpdUser,
???????????????UpdDateTime)
????????????Select CustId,
????????????Case
????????????When( @DeletedCount>0)Then
????????????'Update'
????????Else 'Insert'
??????End,
Current_User,
Current_TimeStamp
From inserted
End
Else If(@DeletedCount>0)Begin
Insert Into AppDta.dbo.CustUpdLog
( CustId,
Action,
UpdUser,
UpdDateTime)
select CustId,
'Delete',
Current_User,
Current_TimeStamp
From deleted
End
??? 正如本例所示,無論何時(shí)Insert 或者Update 語句影響一個(gè)或者多行時(shí),inserted 臨時(shí)表都有記錄行。無論何時(shí)Delete 或者Update 語句影響一個(gè)或者多行時(shí),deleted 臨時(shí)表都有記錄行。對于一個(gè)Update 語句,deleted 臨時(shí)表有舊行,inserted 臨時(shí)表有新行。這個(gè)示例還反映了觸發(fā)器的另一個(gè)重要方面: 對于某個(gè)表的Update 或者Delete 操作,即使該語句沒有影響到行,也激活觸發(fā)器 (也就是說沒有滿足Where 子句的行)。 觸發(fā)器的存儲過程應(yīng)該預(yù)測這種可能性。
??? 不僅可以為一個(gè)表創(chuàng)建多個(gè)觸發(fā)器,而且還可以為一個(gè)表的同一個(gè)SQL 語句(例如Update 語句)創(chuàng)建多個(gè)后觸發(fā)器,不能為同一個(gè)SQL 語句創(chuàng)建多個(gè)前觸發(fā)器。每一個(gè)新的Create Trigger 語句增加觸發(fā)器到那些指定表和語句已有的觸發(fā)器中。對于所創(chuàng)建的多個(gè)觸發(fā)器,可以用系統(tǒng)存儲過程sp_settriggerorder 來指定第一個(gè)被激活的觸發(fā)器和最后一個(gè)被激活的觸發(fā)器,而對于其他的觸發(fā)器,則不能指定其激活順序,只能由系統(tǒng)決定。這種觸發(fā)器的特征不會引起任何特殊的問題。因?yàn)榭偸强梢詫?shí)現(xiàn)各種動作作為正常的存儲過程,并且按照要求的順序從一個(gè)觸發(fā)器中調(diào)用它們。
??? 盡管觸發(fā)器是一種存儲過程,但是不能使用Execute 語句調(diào)用它 ,如果有希望共享觸發(fā)器和正常的存儲過程的編碼,那么只需把共享代碼放在存儲過程中,從觸發(fā)器中調(diào)用它。如果一個(gè)觸發(fā)器修改一個(gè)表,那么這些修改可能會激活另一個(gè)觸發(fā)器,或者本身。在默認(rèn)情況下,SQL Sever 允許這種嵌套的觸發(fā)器調(diào)用深度為32層。雖然我們建議允許嵌套的和疊代的觸發(fā)器,但是可以使用系統(tǒng)存儲過程禁止這么做。下面的語句 在指定的數(shù)據(jù)庫上防止疊代觸發(fā)器:
sp_dboption AppDta,`recursive triggers',`false'
??? 為了 在所有數(shù)據(jù)庫中防止嵌套觸發(fā)器調(diào)用(包括疊代調(diào)用) ,可以使用下面的語句:
sp_configure `nested triggers',0
??? 前面以后觸發(fā)器為例介紹了觸發(fā)器的基本內(nèi)容,下面再介紹一下前觸發(fā)器的不同之處。 要?jiǎng)?chuàng)建一個(gè)前觸發(fā)器必須用Instead Of 顯式聲明 ,如下面的例子:
create Trigger TrackCustomerUpdates
On AppDta.dbo.Customer
Instead Of Update
As
Insert Into AppDta.dbo.CustUpdLog
(CustId,
Action,
UpdUser,
UpdDateTime)
Select CustId,
‘Update’,
Current_User,
Current_TimeStamp
From inserted
??? 與后觸發(fā)器不同的是:前觸發(fā)器既可以在表又可以在視圖上創(chuàng)建,但一條語句只能創(chuàng)建一個(gè)前觸發(fā)器,因此,前觸發(fā)器不存在激活順序問題
觸發(fā)器應(yīng)用舉例:從當(dāng)前數(shù)據(jù)庫服務(wù)器的Shop表Insert操作同步到另一臺服務(wù)器的Shop。
CREATE TRIGGER Trigger_SynShopForInsert1
ON dbo.Shop
FOR INSERT
AS
? insert into OtherServer.dbo.shop
(
lngShopID,strShopCode,strName,strShopName,strDescription,lngIndex
)
select? lngShopID,strShopCode,strName,strShopName,strDescription,lngIndex
from shop where lngShopID in (select lngshopid from inserted)
或者:
CREATE TRIGGER Trigger_SynShopForInsert2
ON dbo.Shop
FOR INSERT
AS
? insert into OtherServer.dbo.shop
(
lngShopID,strShopCode,strName,strShopName,strDescription,lngIndex
)
select? lngShopID,strShopCode,strName,strShopName,strDescription,lngIndex
from? inserted
??? 觸發(fā)器是一種特殊的存儲過程。當(dāng)Insert Update 或者Delete 語句修改表中一個(gè)或者多個(gè)行時(shí)執(zhí)行觸發(fā)器。因?yàn)镾QL Server 對特定表上的每一個(gè)指定操作調(diào)用一個(gè)觸發(fā)器,所以可以使用觸發(fā)器擴(kuò)展SQL Sever 的內(nèi)置完整性和數(shù)據(jù)操縱功能.
??? 注意: 不像Delete 語句,Trancate Table 語句不激活觸發(fā)器,Write Text 語句也不激活觸發(fā)器。
??? 在SQL Sever 2000 中支持兩種類型的觸發(fā)器,前觸發(fā)器(Instead Of Trigger)和后觸發(fā)器(After Trigger)。前觸發(fā)器就是在語句執(zhí)行之前激活觸發(fā)器,而后觸發(fā)器就是在語句執(zhí)行之后激活觸發(fā)器。可以通過FOR 子句來選擇使用何種觸發(fā)器。
??? 當(dāng)為每一種操作創(chuàng)建一個(gè)觸發(fā)器時(shí),可以為所有三種操作創(chuàng)建一個(gè)觸發(fā)器,并且使用相應(yīng)的編程技術(shù)處理每一種操作。下面的示例在For 子句中列出了三種語句類型并,且使用條件語句將相應(yīng)的跟蹤值插入到CustUpdLog 表中。
Create Trigger TrackCustomerUpdates
???On AppDta.dbo.Customer
??????For Insert,Update,Delete
???As
??????Declare @InsertedCount Int
??????Declare @DeletedCount Int
??????Set @InsertedCount=(Select Count(*)From inserted)
??????Set @DeletedCount=(Select Count(*)From deleted)
??????If ( @InsertedCount>0)
?????????Begin
????????????Insert Into AppDta.dbo.CustUpdLog
????????????( CustID,
???????????????Action,
???????????????UpdUser,
???????????????UpdDateTime)
????????????Select CustId,
????????????Case
????????????When( @DeletedCount>0)Then
????????????'Update'
????????Else 'Insert'
??????End,
Current_User,
Current_TimeStamp
From inserted
End
Else If(@DeletedCount>0)Begin
Insert Into AppDta.dbo.CustUpdLog
( CustId,
Action,
UpdUser,
UpdDateTime)
select CustId,
'Delete',
Current_User,
Current_TimeStamp
From deleted
End
??? 正如本例所示,無論何時(shí)Insert 或者Update 語句影響一個(gè)或者多行時(shí),inserted 臨時(shí)表都有記錄行。無論何時(shí)Delete 或者Update 語句影響一個(gè)或者多行時(shí),deleted 臨時(shí)表都有記錄行。對于一個(gè)Update 語句,deleted 臨時(shí)表有舊行,inserted 臨時(shí)表有新行。這個(gè)示例還反映了觸發(fā)器的另一個(gè)重要方面: 對于某個(gè)表的Update 或者Delete 操作,即使該語句沒有影響到行,也激活觸發(fā)器 (也就是說沒有滿足Where 子句的行)。 觸發(fā)器的存儲過程應(yīng)該預(yù)測這種可能性。
??? 不僅可以為一個(gè)表創(chuàng)建多個(gè)觸發(fā)器,而且還可以為一個(gè)表的同一個(gè)SQL 語句(例如Update 語句)創(chuàng)建多個(gè)后觸發(fā)器,不能為同一個(gè)SQL 語句創(chuàng)建多個(gè)前觸發(fā)器。每一個(gè)新的Create Trigger 語句增加觸發(fā)器到那些指定表和語句已有的觸發(fā)器中。對于所創(chuàng)建的多個(gè)觸發(fā)器,可以用系統(tǒng)存儲過程sp_settriggerorder 來指定第一個(gè)被激活的觸發(fā)器和最后一個(gè)被激活的觸發(fā)器,而對于其他的觸發(fā)器,則不能指定其激活順序,只能由系統(tǒng)決定。這種觸發(fā)器的特征不會引起任何特殊的問題。因?yàn)榭偸强梢詫?shí)現(xiàn)各種動作作為正常的存儲過程,并且按照要求的順序從一個(gè)觸發(fā)器中調(diào)用它們。
??? 盡管觸發(fā)器是一種存儲過程,但是不能使用Execute 語句調(diào)用它 ,如果有希望共享觸發(fā)器和正常的存儲過程的編碼,那么只需把共享代碼放在存儲過程中,從觸發(fā)器中調(diào)用它。如果一個(gè)觸發(fā)器修改一個(gè)表,那么這些修改可能會激活另一個(gè)觸發(fā)器,或者本身。在默認(rèn)情況下,SQL Sever 允許這種嵌套的觸發(fā)器調(diào)用深度為32層。雖然我們建議允許嵌套的和疊代的觸發(fā)器,但是可以使用系統(tǒng)存儲過程禁止這么做。下面的語句 在指定的數(shù)據(jù)庫上防止疊代觸發(fā)器:
sp_dboption AppDta,`recursive triggers',`false'
??? 為了 在所有數(shù)據(jù)庫中防止嵌套觸發(fā)器調(diào)用(包括疊代調(diào)用) ,可以使用下面的語句:
sp_configure `nested triggers',0
??? 前面以后觸發(fā)器為例介紹了觸發(fā)器的基本內(nèi)容,下面再介紹一下前觸發(fā)器的不同之處。 要?jiǎng)?chuàng)建一個(gè)前觸發(fā)器必須用Instead Of 顯式聲明 ,如下面的例子:
create Trigger TrackCustomerUpdates
On AppDta.dbo.Customer
Instead Of Update
As
Insert Into AppDta.dbo.CustUpdLog
(CustId,
Action,
UpdUser,
UpdDateTime)
Select CustId,
‘Update’,
Current_User,
Current_TimeStamp
From inserted
??? 與后觸發(fā)器不同的是:前觸發(fā)器既可以在表又可以在視圖上創(chuàng)建,但一條語句只能創(chuàng)建一個(gè)前觸發(fā)器,因此,前觸發(fā)器不存在激活順序問題
觸發(fā)器應(yīng)用舉例:從當(dāng)前數(shù)據(jù)庫服務(wù)器的Shop表Insert操作同步到另一臺服務(wù)器的Shop。
CREATE TRIGGER Trigger_SynShopForInsert1
ON dbo.Shop
FOR INSERT
AS
? insert into OtherServer.dbo.shop
(
lngShopID,strShopCode,strName,strShopName,strDescription,lngIndex
)
select? lngShopID,strShopCode,strName,strShopName,strDescription,lngIndex
from shop where lngShopID in (select lngshopid from inserted)
或者:
CREATE TRIGGER Trigger_SynShopForInsert2
ON dbo.Shop
FOR INSERT
AS
? insert into OtherServer.dbo.shop
(
lngShopID,strShopCode,strName,strShopName,strDescription,lngIndex
)
select? lngShopID,strShopCode,strName,strShopName,strDescription,lngIndex
from? inserted
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯(lián)系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長非常感激您!手機(jī)微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
