溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何在mysql存儲(chǔ)過程中使用游標(biāo)

發(fā)布時(shí)間:2021-05-26 09:44:19 來源:億速云 閱讀:467 作者:Leah 欄目:MySQL數(shù)據(jù)庫(kù)

本篇文章給大家分享的是有關(guān)如何在mysql存儲(chǔ)過程中使用游標(biāo),小編覺得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

在處理存儲(chǔ)過程中的結(jié)果集時(shí),可以使用游標(biāo),因?yàn)橛螛?biāo)允許我們迭代查詢返回的一組行,并相應(yīng)地處理每行。mysql的游標(biāo)為只讀,不可滾動(dòng)和敏感三種模式,我們來看下:

  • 只讀:無法通過光標(biāo)更新基礎(chǔ)表中的數(shù)據(jù)。

  • 不可滾動(dòng):只能按照select語句確定的順序獲取行。不能以相反的順序獲取行。 此外,不能跳過行或跳轉(zhuǎn)到結(jié)果集中的特定行。

  • 敏感:有兩種游標(biāo):敏感游標(biāo)和不敏感游標(biāo)。敏感游標(biāo)指向?qū)嶋H數(shù)據(jù),不敏感游標(biāo)使用數(shù)據(jù)的臨時(shí)副本。敏感游標(biāo)比一個(gè)不敏感的游標(biāo)執(zhí)行得更快,因?yàn)樗恍枰R時(shí)拷貝數(shù)據(jù)。但是,對(duì)其他連接的數(shù)據(jù)所做的任何更改都將影響由敏感游標(biāo)使用的數(shù)據(jù),因此,如果不更新敏感游標(biāo)所使用的數(shù)據(jù),則更安全。 MySQL游標(biāo)是敏感的。

我們可以在存儲(chǔ)過程,存儲(chǔ)函數(shù)和觸發(fā)器中使用MySQL游標(biāo),咱們先來看下使用DECLARE語句聲明的語法:

DECLARE cursor_name CURSOR FOR SELECT_statement;

我們要注意下,游標(biāo)聲明必須在變量聲明之后。如果在變量聲明之前聲明游標(biāo),mysql將會(huì)發(fā)出一個(gè)錯(cuò)誤,另外游標(biāo)必須始終與SELECT語句相關(guān)聯(lián)。完事呢,我們來使用OPEN語句打開游標(biāo)。OPEN語句初始化游標(biāo)的結(jié)果集,因此我們必須在從結(jié)果集中提取行之前調(diào)用OPEN語句:

OPEN cursor_name;

然后,我們來使用FETCH語句來檢索光標(biāo)指向的下一行,并將光標(biāo)移動(dòng)到結(jié)果集中的下一行:

FETCH cursor_name INTO variables list;

之后,我們就可以檢查是否有任何行記錄可用,然后再提取它。完事最后還要記得調(diào)用CLOSE語句來停用光標(biāo)并釋放與之關(guān)聯(lián)的內(nèi)存:

CLOSE cursor_name;

我們要知道,當(dāng)光標(biāo)不再使用時(shí),應(yīng)該關(guān)閉它。當(dāng)我們使用mysql游標(biāo)時(shí),還必須聲明一個(gè)NOT FOUND處理程序來處理當(dāng)游標(biāo)找不到任何行時(shí)的情況。 因?yàn)槊看握{(diào)用FETCH語句時(shí),游標(biāo)會(huì)嘗試讀取結(jié)果集中的下一行。 當(dāng)光標(biāo)到達(dá)結(jié)果集的末尾時(shí),它將無法獲得數(shù)據(jù),并且會(huì)產(chǎn)生一個(gè)條件。NOT FOUND處理程序用于處理這種情況,我們來看下它的語法結(jié)構(gòu):

DECLARE CONTINUE HANDLER FOR NOT FOUND SET finished = 1;

finished是一個(gè)變量,指示光標(biāo)到達(dá)結(jié)果集的結(jié)尾。請(qǐng)注意,處理程序聲明必須出現(xiàn)在存儲(chǔ)過程中的變量和游標(biāo)聲明之后。我們?cè)賮砜聪耺ysql游標(biāo)的運(yùn)行原理圖:

如何在mysql存儲(chǔ)過程中使用游標(biāo)

咱們接下來,就要開發(fā)一個(gè)存儲(chǔ)過程用來獲取employees表中所有員工的電子郵件列表。我們先來聲明一些變量,一個(gè)用于循環(huán)員工電子郵件的游標(biāo)和一個(gè)NOT FOUND處理程序:

DECLARE finished INTEGER DEFAULT 0;
DECLARE email varchar(255) DEFAULT "";
-- declare cursor for employee email
DEClARE email_cursor CURSOR FOR
 SELECT email FROM employees;
-- declare NOT FOUND handler
DECLARE CONTINUE HANDLER
FOR NOT FOUND SET finished = 1;
接下來,使用OPEN語句打開email_cursor:
OPEN email_cursor;

然后,迭代電子郵件列表,并使用分隔符(;)連接每個(gè)電子郵件:

get_email: LOOP
 FETCH email_cursor INTO v_email;
 IF v_finished = 1 THEN
 LEAVE get_email;
 END IF;
 -- build email list
 SET email_list = CONCAT(v_email,";",email_list);
END LOOP get_email;

之后,我們?cè)谘h(huán)中,使用v_finished變量來檢查列表中是否有任何電子郵件來終止循環(huán),完事呢,使用CLOSE語句關(guān)閉游標(biāo):

CLOSE email_cursor;

我們來看下build_email_list存儲(chǔ)過程所有代碼:

DELIMITER $$
CREATE PROCEDURE build_email_list (INOUT email_list varchar(4000))
BEGIN
 DECLARE v_finished INTEGER DEFAULT 0;
    DECLARE v_email varchar(100) DEFAULT "";
 -- declare cursor for employee email
 DEClARE email_cursor CURSOR FOR
 SELECT email FROM employees;
 -- declare NOT FOUND handler
 DECLARE CONTINUE HANDLER
    FOR NOT FOUND SET v_finished = 1;
 OPEN email_cursor;
 get_email: LOOP
 FETCH email_cursor INTO v_email;
 IF v_finished = 1 THEN
 LEAVE get_email;
 END IF;
 -- build email list
 SET email_list = CONCAT(v_email,";",email_list);
 END LOOP get_email;
 CLOSE email_cursor;
END$$
DELIMITER ;

我們來使用以下腳本測(cè)試build_email_list存儲(chǔ)過程:

SET @email_list = "";
CALL build_email_list(@email_list);
SELECT @email_list;

以上就是如何在mysql存儲(chǔ)過程中使用游標(biāo),小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見到或用到的。希望你能通過這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI