mysql觸發器old實例(mysql觸發器old和new)

本文目錄一覽:

關於mysql中觸發器old和new如何更好的區別我有話要說

Mysql的觸發器相當於內部處理的一些過程,不帶入和帶出任何的參數。

其內部使用的參數就是新舊兩條記錄old和new的字段。

用於完成數據表之間的觸發操作,來保證數據庫的一致性、完整性。

Mysql的存儲過程是類似於其它編程語言中的函數的功能。

存儲過程內部可以使用順序循環和轉移三種基本程序結構,而且整個存儲過程可以接受和返回參數。

數據庫的觸發器怎麼進行高級操作?

1.new與old在數據庫的觸發器中經常會用到更新前的值和更新後的值,所以掌握new和old的語法很重要。

old:表示操作執行前的數據行。

new:表示操作執行後的數據行。

一User數據表如圖User數據表所示,若執行以下更新操作語句:updateUsersetscore=80whereuser_id=1

User數據表

則在此操作中,old表示未執行update語句前user_id=1這行記錄;而new表示執行update語句後user_id=1這行記錄。

從上面的表述中可知,new與old均表示某一行記錄,old所表示的是寫操作發生前的這一行舊數據,new則表示寫操作發生後的這一行新的數據。正因如此,可以把new與old看作面向對象編程裏面的一個對象或實例,與面向對象的方式類似,可用new.字段名或old.字段名的方式進行存取值。

old.字段名:表示未執行操作前的該行對應的某字段值。

new.字段名:表示執行操作後的該行對應的某字段值。

在上面User表的update操作中:old.score=60,表示update操作前score字段的舊值是60。

new.score=80,表示update操作後score字段的新值是80。

如果要使用new語句進行賦值,只能在before類型的觸發器中使用,不能在after類型的觸發器中使用。

更新操作前使用before先賦值,再插入數據庫中。如以下語句是正確的:CREATETRIGGERupdatepriceBEFOREinsertONconsumeinfoFOREACHROWBEGINsetnew.金額=0;END更新操作後,不能在after中用new賦值,因為操作已經結束,只能讀取內容。如以下語句是錯誤的:CEATETRIGGERupdatepriceAFTERinsertONconsumeinfoFOREACHROWBEGINsetnew.金額=0;ENDnew與old的區別:前者可在before觸發器中賦值、取值,也可在after觸發器中取值;

後者只能用於取值,因為賦值沒有意義。

注意:INSERT語句,只有new合法;

DELETE語句,只有old才合法;

UPDATE語句,可以同時使用new和old。

2.before與afterbefore與after表示觸發器觸發的時間點是在寫操作開始之前,還是在寫操作完成後,正因為它們有時間點先後的問題,因此它們的功能與使用場合有非常大的差別。

(1)before((1)先完成觸發操作,再執行業務數據的增刪改。

(2)觸發的語句先於監視的業務語句。

(3)有機會影響即將發生的操作。

2)after((1)先完成業務數據的增刪改,再觸發。

(2)觸發的語句晚於監視的業務語句。

(3)無法影響前面的增刪改動作。

3.異常處理MySQL現有版本中不支持自定義異常,當某處需要拋出異常時,可拋出一個系統異常(類似運行異常)。如故意往不存在的表中插入數據等方式來觸發系統異常的拋出,當異常拋出時,本次正在執行的所有操作會終止執行,並回滾所有數據到操作發生前的狀態。

在觸發器中,需要拋出異常的場景有很多,以下兩個方面較為普遍:((1)新進來的數據不符合業務邏輯。

例如,①倉庫最大商品庫存數為10,訂單要求一次性購買數量20。

②銀行賬戶上只有50元,想要支付100元的賬單。

(2)權限不足,不允許操作。

例如,①普通用戶通過非法途徑操作核心資源表。

②在非工作日修改業務數據。4.觸發器案例用mytab.sql腳本創建表環境,然後用下面語句創建觸發器。當往mytab表添加記錄時,觸發器中將拋出異常,導致所有操作終止,並回滾所有操作的數據。

DELIMITER//DROPTRIGGERIFEXISTSt_exception_trigger;CREATETRIGGERt_exception_triggerAFTERINSERTONmytabFOREACHROWBEGIN–tab3表不存在

計算機mysql觸發器中創建觸發器裏面監視的table和下面的new或者是old指代的列有什麼關係?

建立兩個單域的表格。一個表格中為姓名列表(表格名:data)。

另一個表格中是所插入字符的字符數(表格名:chars)。在data表格中定義一個觸發器。

每次在其中插入一個新姓名時,chars表格中運行的總數就會根據新插入記錄的字符數目進行自動更新。

(見列表A)

mysql CREATE TABLE data (name VARCHAR(255));

Query OK, 0 rows affected (0.09 sec)

mysql CREATE TABLE chars (count INT(10));

Query OK, 0 rows affected (0.07 sec)

mysql INSERT INTO chars (count) VALUES (0);

Query OK, 1 row affected (0.00 sec)

mysql CREATE TRIGGER t1 AFTER INSERT ON

data FOR EACH ROW UPDATE chars SET count = count + CHAR_LENGTH(NEW.name);

Query OK, 0 rows affected (0.01 sec)

列表A

理解上面代碼的關鍵在於CREATE TRIGGER命令,被用來定義一個新觸發器。這個命令建立一個新觸發器,假定的名稱為t1,每次有一個新記錄插入到data表格中時,t1就被激活。

在這個觸發器中有兩個重要的子句:

AFTER INSERT子句表明觸發器在新記錄插入data表格後激活。

UPDATE chars SET count = count + CHAR_LENGTH(NEW.name)子句表示觸發器激活後執行的SQL命令。在本例中,該命令表明用新插入的data.name域的字符數來更新 chars.count欄。這一信息可通過內置的MySQL函數CHAR_LENGTH()獲得。

放在源表格域名前面的NEW關鍵字也值得注意。這個關鍵字表明觸發器應考慮域的new值(也就是說,剛被插入到域中的值)。MySQL還支持相應的OLD前綴,可用它來指域以前的值。

可以通過調用SHOW TRIGGER命令來檢查觸發器是否被激活,如列表B所示。

mysql SHOW TRIGGERS\G

*************************** 1. row ***************************

?Trigger: t1

?Event: INSERT

?Table: data

Statement: UPDATE chars SET count = count + CHAR_LENGTH(NEW.name)

Timing: AFTER

?Created: NULL

ql_mode:

1 row in set (0.01 sec)

列表B

激活觸發器後,開始對它進行測試。試着在data表格中插入幾個記錄:

mysql INSERT INTO data (name) VALUES (‘Sue’), (‘Jane’);

Query OK, 2 rows affected (0.00 sec)

Records: 2?Duplicates: 0?Warnings: 0

然後檢查chars表格看觸發器是否完成它該完成的任務:

mysql SELECT * FROM chars;

+——-+

| count |

+——-+

| 7|

+——-+

1 row in set (0.00 sec)

data表格中的INSERT命令激活觸發器,計算插入記錄的字符數,並將結果存儲在chars表格中。如果往data表格中增加另外的記錄,chars.count值也會相應增加。

觸發器應用完畢後,可有DROP TRIGGER命令輕鬆刪除它。

mysql DROP TRIGGER t1;

Query OK, 0 rows affected (0.00 sec)

注意:理想情況下,你還需要一個倒轉觸發器,每當一個記錄從源表格中刪除時,它從字符總數中減去記錄的字符數。這很容易做到,你可以把它當作練習來完成。提示:應用BEFORE DELETE ON子句是其中一種方法。

現在,要建立一個審計記錄來追蹤對這個表格所做的改變。這個記錄將反映表格的每項改變,並向用戶說明由誰做出改變以及改變的時間。需要建立一個新表格來存儲這一信息(表格名:audit),如下所示。(列表C)

mysql CREATE TABLE audit (id INT(7), balance FLOAT, user VARCHAR(50)

NOT NULL, time TIMESTAMP NOT NULL);

Query OK, 0 rows affected (0.09 sec)

列表C

接下來,我將在accounts表格中定義一個觸發器。(列表D)

mysql CREATE TRIGGER t1 AFTER UPDATEON accounts

FOR EACH ROW INSERT INTO audit (id, balance, user, time)

VALUES (OLD.id, NEW.balance, CURRENT_USER(), NOW());

Query OK, 0 rows affected (0.04 sec)

列表D

要是已經走到這一步,就很容易理解。accounts表格每經歷一次UPDATE,觸發器插入(INSERT)對應記錄的id、新的餘額、當前時間和登錄audit表格的用戶的名稱。

實現中的例子:用觸發器審計記錄

既然了觸發器的基本原理,來看一個稍稍複雜的例子。常用觸發器來建立一個自動「審計記錄」,以記錄各種用戶對數據庫的更改。為了解審計記錄的實際應用,請看下面的表格(表格名:accounts),它列出了一個用戶的三個銀行賬戶餘額。(表A)

mysql SELECT * FROM accounts;

+—-+————+———+

| id | label| balance |

+—-+————+———+

|1 | Savings #1 |500 |

|2 | Current #1 |2000 |

|3 | Current #2 |3500 |

+—-+————+———+

3 rows in set (0.00 sec)

表A

然後,檢查觸發器是否被激活:

mysql SHOW TRIGGERS \G

*************************** 1. row ***************************

?Trigger: t1

?Event: UPDATE

?Table: accounts

Statement: INSERT INTO audit (id, balance, user, time)

VALUES (OLD.id, NEW.balance, CURRENT_USER(), NOW())

Timing: AFTER

?Created: NULL

Sql_mode:

1 row in set (0.01 sec)

再來看最後的結果(列表E):

mysql UPDATE accounts SET balance = 500 WHERE id = 1;

Query OK, 1 row affected (0.00 sec)

Rows matched: 1?Changed: 1?Warnings: 0

mysql UPDATE accounts SET balance = 900 WHERE id = 3;

Query OK, 1 row affected (0.01 sec)

Rows matched: 1?Changed: 1?Warnings: 0

mysql UPDATE accounts SET balance = 1900 WHERE id = 1;

Query OK, 1 row affected (0.00 sec)

Rows matched: 1?Changed: 1?Warnings: 0

列表E

注意,對accounts表格所作的改變已被記錄到audit表格中,將來如果出現問題,可以方便地從中進行恢復。

mysql SELECT * FROM audit;

+——+———+—————-+———————+

| id| balance | user| time|

+——+———+—————-+———————+

|1 |500 | root@localhost | 2006-04-22 12:52:15 |

|3 |900 | root@localhost | 2006-04-22 12:53:15 |

|1 |1900 | root@localhost | 2006-04-22 12:53:23 |

+——+———+—————-+———————+

3 rows in set (0.00 sec)

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/302947.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-12-31 11:48
下一篇 2024-12-31 11:48

相關推薦

  • 如何修改mysql的端口號

    本文將介紹如何修改mysql的端口號,方便開發者根據實際需求配置對應端口號。 一、為什麼需要修改mysql端口號 默認情況下,mysql使用的端口號是3306。在某些情況下,我們需…

    編程 2025-04-29
  • Python中new和init的區別

    new和init都是Python中常用的魔法方法,它們分別負責對象的創建和初始化,本文將從多個角度詳細闡述它們的區別。 一、創建對象 new方法是用來創建一個對象的,它是一個類級別…

    編程 2025-04-29
  • Python生成隨機數的應用和實例

    本文將向您介紹如何使用Python生成50個60到100之間的隨機數,並將列舉使用隨機數的幾個實際應用場景。 一、生成隨機數的代碼示例 import random # 生成50個6…

    編程 2025-04-29
  • Python操作MySQL

    本文將從以下幾個方面對Python操作MySQL進行詳細闡述: 一、連接MySQL數據庫 在使用Python操作MySQL之前,我們需要先連接MySQL數據庫。在Python中,我…

    編程 2025-04-29
  • MySQL遞歸函數的用法

    本文將從多個方面對MySQL遞歸函數的用法做詳細的闡述,包括函數的定義、使用方法、示例及注意事項。 一、遞歸函數的定義 遞歸函數是指在函數內部調用自身的函數。MySQL提供了CRE…

    編程 2025-04-29
  • MySQL bigint與long的區別

    本文將從數據類型定義、存儲空間、數據範圍、計算效率、應用場景五個方面詳細闡述MySQL bigint與long的區別。 一、數據類型定義 bigint在MySQL中是一種有符號的整…

    編程 2025-04-28
  • 如何在dolphinscheduler中運行chunjun任務實例

    本文將從多個方面對dolphinscheduler運行chunjun任務實例進行詳細的闡述,包括準備工作、chunjun任務配置、運行結果等方面。 一、準備工作 在運行chunju…

    編程 2025-04-28
  • MySQL左連接索引不生效問題解決

    在MySQL數據庫中,經常會使用左連接查詢操作,但是左連接查詢中索引不生效的情況也比較常見。本文將從多個方面探討MySQL左連接索引不生效問題,並給出相應的解決方法。 一、索引的作…

    編程 2025-04-28
  • JavaScript中使用new Date轉換為YYYYMMDD格式

    在JavaScript中,我們通常會使用Date對象來表示日期和時間。當我們需要在網站上顯示日期時,很多情況下需要將Date對象轉換成YYYYMMDD格式的字符串。下面我們來詳細了…

    編程 2025-04-27
  • CentOS 7在線安裝MySQL 8

    在本文中,我們將介紹如何在CentOS 7操作系統中在線安裝MySQL 8。我們會從安裝環境的準備開始,到安裝MySQL 8的過程進行詳細的闡述。 一、環境準備 在進行MySQL …

    編程 2025-04-27

發表回復

登錄後才能評論