資料庫 觸發器

来源:https://www.cnblogs.com/mingqi-420/archive/2019/04/20/10740301.html
-Advertisement-
Play Games

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


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

as

              delete 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

as

begin

    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 資料庫名

             go

             select * from sysobjects where xtype='TR'

sysobjects 保存著資料庫的對象,其中 xtype 為 TR 的記錄即為觸發器對象。在 name 一列,我們可以看到觸發器名稱。

 

(2).sp_helptext 查看觸發器內容

 

use 資料庫名

 

go

 

exec sp_helptext '觸發器名稱'

 

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

 

 除了觸發器外,sp_helptext 還可以顯示 規則、預設值、未加密的存儲過程、用戶定義函數、視圖的文本。

 

(3).sp_helptrigger 用於查看觸發器的屬性

 

  sp_helptrigger 有兩個參數:第一個參數為表名;第二個為觸發器類型,為 char(6) 類型,可以是 INSERT、UPDATE、DELETE,如果省略則顯示指定表中所有類型觸發器的屬性。

 

use 資料庫名

 

go

 

exec sp_helptrigger tableName

 

2.禁用啟用觸發器

 

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

 

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

 

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

 

  如果把“觸發器名稱”換成“ALL”,則表示禁用或啟用該表的全部觸發器。

 

3修改觸發器

 

--修改觸發器語法

 

ALTER TRIGGER  trigger_name

 

     ON  table_name

 

     [ WITH ENCRYPTION ]

 

     FOR {[DELETE][,][INSERT][,][UPDATE]}

 

     AS

 

       sql_statement;

 

4.刪除觸發器

 

 --語法格式:

 

      DROP  TRIGGER   { trigger } [ ,...n ]

 

參數:

 

 trigger: 要刪除的觸發器名稱

 

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

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 筆記記錄自林曉斌(丁奇)老師的《MySQL實戰45講》 (本篇內圖片均來自丁奇老師的講解,如有侵權,請聯繫我刪除) 19) --為什麼我只查一行的語句,也執行這麼慢? 需要說明一下,如果MySQL資料庫本身就有很大的壓力,導致資料庫伺服器CPU占用率很高或ioutil(IO利用率)很高,這種情況下所 ...
  • 持續關註MongoDB博客(https://www.mongodb.com/blog)的同學一定會留意到,技術大牛Daniel Coupal 和 Ken W. Alger ,從 今年 2月17 號開始,在博客上持續發表了 如何在MongoDB中設計資料庫模式的方法。截止到今日(4月20號),12種模 ...
  • 由於本人最近已經成為TiDB的粉絲,所以就開始各種研究TiDB的源碼,研究源碼這個事情,首先就需要在自己電腦上不斷的調試及修改。TiDB本身的代碼是非常容易編譯和調試的,但是要把PD、TiKV集群同時在本機上建立起來,還是有一點難度的。好在pingcap官方提供了docker-compose搭建集群... ...
  • 一、問題現象:在對資料庫進行expdp導出時發生報錯ora-16000,腳本如下: 報錯如下: 由上報錯可知,資料庫read—only,查看資料庫狀態: 果然,資料庫是只讀狀態。 二、expdp研究 將資料庫設置為讀寫狀態,觀察expdp 重新執行expdp導出腳本,觀察執行情況 通過plsql查看 ...
  • sqli labs 下載、安裝 下載地址:https://github.com/Audi 1/sqli labs phpstudy:http://down.php.cn/PhpStudy20180211.zip 所需安裝環境支持包:http://www.pc6.com/softview/SoftVi ...
  • --constraint--not null 非空約束create table demo01(empno number(4),ename varchar2(10) not null,job varchar2(10))insert into demo01 values(1234,' ','SALES' ...
  • 寫了兩年的php,一直依賴於phpstorm,其他三方工具和框架集成的高效語句,造成的後果就是基礎愈發陌生,手寫原生sql和php代碼等卡殼嚴重。亡羊補牢!ps:原文是markdown,沒修改格式直接傳了 ...
  • 近期在給客戶做新數據交換方案調試時發現一處視圖創建語句帶不出數據。 精簡需求後如下:a部門從b部門獲取主體數據,由於a、b兩部門有些代碼標準不一致需要做轉換。於是開發寫了個對照表做轉換生成業務視圖。 主表zb數據如下,B_MD1和B_DM2是兩種類型代碼,分別是lx1和lx2(比如一個是證件類型代碼 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...