本文實(shí)例講述了mysql引發(fā)存儲(chǔ)過(guò)程中的錯(cuò)誤條件(SIGNAL和RESIGNAL語(yǔ)句)。分享給大家供大家參考,具體如下:
在mysql中,我們可以使用SIGNAL和RESIGNAL語(yǔ)句來(lái)引發(fā)存儲(chǔ)過(guò)程中的錯(cuò)誤條件。
先來(lái)看,SIGNAL語(yǔ)句。我們通常使用SIGNAL語(yǔ)句在存儲(chǔ)的程序(例如存儲(chǔ)過(guò)程,存儲(chǔ)函數(shù),觸發(fā)器或事件)中向調(diào)用者返回錯(cuò)誤或警告條件。 SIGNAL語(yǔ)句提供了對(duì)返回值(如值和消息SQLSTATE)的信息的控制。來(lái)看下它的語(yǔ)法結(jié)構(gòu):
SIGNAL SQLSTATE | condition_name;
SET condition_information_item_name_1 = value_1,
condition_information_item_name_1 = value_2, etc;
SIGNAL關(guān)鍵字是由DECLARE CONDITION語(yǔ)句聲明的SQLSTATE值或條件名稱(chēng)。不過(guò)要注意的是,SIGNAL語(yǔ)句必須始終指定使用SQLSTATE值定義的SQLSTATE值或命名條件。完事我們?nèi)绻蛘{(diào)用者提供信息,就得使用SET子句,如果要使用值返回多個(gè)條件信息項(xiàng)名稱(chēng),則需要用逗號(hào)分隔每個(gè)名稱(chēng)/值對(duì)。上述sql中,condition_information_item_name可以是MESSAGE_TEXT,MYSQL_ERRORNO,CURSOR_NAME等。咱們來(lái)看一個(gè)將訂單行項(xiàng)目添加到現(xiàn)有銷(xiāo)售訂單中的存儲(chǔ)過(guò)程,如果訂單號(hào)碼不存在,它會(huì)發(fā)出錯(cuò)誤消息:
DELIMITER $$
CREATE PROCEDURE AddOrderItem(in orderNo int,
in productCode varchar(45),
in qty int,in price double, in lineNo int )
BEGIN
DECLARE C INT;
SELECT COUNT(orderNumber) INTO C
FROM orders
WHERE orderNumber = orderNo;
-- check if orderNumber exists
IF(C != 1) THEN
SIGNAL SQLSTATE '45000'
SET MESSAGE_TEXT = 'Order No not found in orders table';
END IF;
-- more code below
-- ...
END $$
DELIMITER ;
一開(kāi)始,它使用傳遞給存儲(chǔ)過(guò)程的輸入訂單號(hào)對(duì)訂單進(jìn)行計(jì)數(shù),完事如果訂單數(shù)不是1,它會(huì)引發(fā)SQLSTATE 45000的錯(cuò)誤以及orders表中不存在訂單號(hào)的錯(cuò)誤消息。其中45000是一個(gè)通用SQLSTATE值,用于說(shuō)明未處理的用戶定義異常。
我們來(lái)調(diào)用存儲(chǔ)過(guò)程AddOrderItem(),但是傳遞不存在的訂單號(hào),那么將收到一條錯(cuò)誤消息:
CALL AddOrderItem(10,'S10_1678',1,95.7,1);
執(zhí)行上面代碼,得到以下結(jié)果:
mysql> CALL AddOrderItem(10,'S10_1678',1,95.7,1);
1644 - Order No not found in orders table
mysql>
咱們?cè)賮?lái)看RESIGNAL語(yǔ)句。它在功能和語(yǔ)法方面與SIGNAL語(yǔ)句相似,只是有以下區(qū)別:
- 必須在錯(cuò)誤或警告處理程序中使用RESIGNAL語(yǔ)句,否則您將收到一條錯(cuò)誤消息,指出“RESIGNAL when handler is not active”。 請(qǐng)注意,您可以在存儲(chǔ)過(guò)程中的任何位置使用SIGNAL語(yǔ)句。
- 可以省略RESIGNAL語(yǔ)句的所有屬性,甚至可以省略SQLSTATE值。
如果單獨(dú)使用RESIGNAL語(yǔ)句,則所有屬性與傳遞給條件處理程序的屬性相同。咱們來(lái)看一個(gè)在將發(fā)送給調(diào)用者之前更改錯(cuò)誤消息的存儲(chǔ)過(guò)程:
DELIMITER $$
CREATE PROCEDURE Divide(IN numerator INT, IN denominator INT, OUT result double)
BEGIN
DECLARE division_by_zero CONDITION FOR SQLSTATE '22012';
DECLARE CONTINUE HANDLER FOR division_by_zero
RESIGNAL SET MESSAGE_TEXT = 'Division by zero / Denominator cannot be zero';
--
IF denominator = 0 THEN
SIGNAL division_by_zero;
ELSE
SET result := numerator / denominator;
END IF;
END $$
DELIMITER ;
然后,我們來(lái)嘗試調(diào)用:
mysql> CALL Divide(10,0,@result);
1644 - Division by zero / Denominator cannot be zero
好啦,本次記錄就到這里了,不知道大家有沒(méi)有什么收獲。
更多關(guān)于MySQL相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《MySQL存儲(chǔ)過(guò)程技巧大全》、《MySQL常用函數(shù)大匯總》、《MySQL日志操作技巧大全》、《MySQL事務(wù)操作技巧匯總》及《MySQL數(shù)據(jù)庫(kù)鎖相關(guān)技巧匯總》
希望本文所述對(duì)大家MySQL數(shù)據(jù)庫(kù)計(jì)有所幫助。
您可能感興趣的文章:- MySQL中的if和case語(yǔ)句使用總結(jié)
- mysql存儲(chǔ)過(guò)程之游標(biāo)(DECLARE)原理與用法詳解
- mysql存儲(chǔ)過(guò)程之返回多個(gè)值的方法示例
- mysql存儲(chǔ)過(guò)程之創(chuàng)建(CREATE PROCEDURE)和調(diào)用(CALL)及變量創(chuàng)建(DECLARE)和賦值(SET)操作方法
- mysql存儲(chǔ)過(guò)程之錯(cuò)誤處理實(shí)例詳解
- mysql存儲(chǔ)過(guò)程原理與使用方法詳解
- mysql 存儲(chǔ)過(guò)程中變量的定義與賦值操作
- mysql存儲(chǔ)過(guò)程 游標(biāo) 循環(huán)使用介紹
- MySQL存儲(chǔ)過(guò)程例子(包含事務(wù),輸出參數(shù),嵌套調(diào)用)
- MySql存儲(chǔ)過(guò)程與函數(shù)詳解
- mysql存儲(chǔ)過(guò)程之if語(yǔ)句用法實(shí)例詳解