本文介紹什麼是 SQL 游標,為什麼使用游標,如何使用游標。你使用的 DBMS 可能會提供某種形式的游標,以及這裡沒有提及的功能。更詳細的內容請參閱具體的 DBMS 文檔。 一、游標 SQL 檢索操作返回一組稱為結果集的行,這組返回的行都是與 SQL 語句相匹配的行(零行到多行)。 簡單地使用 SE ...
目錄
本文介紹什麼是 SQL 游標,為什麼使用游標,如何使用游標。你使用的 DBMS 可能會提供某種形式的游標,以及這裡沒有提及的功能。更詳細的內容請參閱具體的 DBMS 文檔。
一、游標
SQL 檢索操作返回一組稱為結果集的行,這組返回的行都是與 SQL 語句相匹配的行(零行到多行)。
簡單地使用 SELECT 語句,沒有辦法得到第一行、下一行或前 10 行。但這是關係 DBMS 功能的組成部分。
結果集(result set)
SQL 查詢所檢索出的結果。
有時,需要在檢索出來的行中前進或後退一行或多行,這就是游標的用途所在。
游標(cursor)是一個存儲在 DBMS 伺服器上的資料庫查詢,它不是一條 SELECT
語句,而是被該語句檢索出來的結果集。
在存儲了游標之後,應用程式可以根據需要滾動或瀏覽其中的數據。
說明:SQLite 支持
SQLite 支持的游標稱為步驟(step),本文講述的基本概念適用於 SQLite 的步驟,但語法可能完全不同。
不同的 DBMS 支持不同的游標選項和特性。常見的一些選項和特性如下。
- 能夠標記游標為只讀,使數據能讀取,但不能更新和刪除。
- 能控制可以執行的定向操作(向前、向後、第一、最後、絕對位置和相對位置等)。
- 能標記某些列為可編輯的,某些列為不可編輯的。
- 規定範圍,使游標對創建它的特定請求(如存儲過程)或對所有請求可訪問。
- 指示 DBMS 對檢索出的數據(而不是指出表中活動數據)進行複製,使數據在游標打開和訪問期間不變化。
游標主要用於互動式應用,其中用戶需要滾動屏幕上的數據,並對數據進行瀏覽或做出更改。
二、使用游標
使用游標涉及幾個明確的步驟。
- 在使用游標前,必須聲明(定義)它。這個過程實際上沒有檢索數據,它只是定義要使用的
SELECT
語句和游標選項。 - 一旦聲明,就必須打開游標以供使用。這個過程用前面定義的
SELECT
語句把數據實際檢索出來。 - 對於填有數據的游標,根據需要取出(檢索)各行。
- 在結束游標使用時,必須關閉游標,可能的話,釋放游標(有賴於具體的 DBMS)。
聲明游標後,可根據需要頻繁地打開和關閉游標。在游標打開時,可根據需要頻繁地執行取操作。
2.1 創建游標
使用 DECLARE
語句創建游標,這條語句在不同的 DBMS 中有所不同。
DECLARE
命名游標,並定義相應的 SELECT
語句,根據需要帶 WHERE
和其他子句。
為了說明,我們創建一個游標來檢索沒有電子郵件地址的所有顧客,作為應用程式的組成部分,幫助操作人員找出空缺的電子郵件地址。
下麵是創建此游標的 DB2、MariaDB、MySQL 和 SQL Server 版本。
DECLARE CustCursor CURSOR
FOR
SELECT * FROM Customers
WHERE cust_email IS NULL;
下麵是 Oracle 和 PostgreSQL 版本:
DECLARE CURSOR CustCursor
IS
SELECT * FROM Customers
WHERE cust_email IS NULL;
在上面兩個版本中,DECLARE
語句用來定義和命名游標,這裡為 CustCursor
。SELECT
語句定義一個包含沒有電子郵件地址(NULL
值)的所有顧客的游標。
定義游標之後,就可以打開它了。
2.2 使用游標
使用 OPEN CURSOR
語句打開游標,這條語句很簡單,在大多數 DBMS 中的語法相同:
OPEN CURSOR CustCursor
在處理 OPEN CURSOR
語句時,執行查詢,存儲檢索出的數據以供瀏覽和滾動。
現在可以用 FETCH
語句訪問游標數據了。FETCH
指出要檢索哪些行,從何處檢索它們以及將它們放於何處(如變數名)。
第一個例子使用 Oracle 語法從游標中檢索一行(第一行):
DECLARE TYPE CustCursor IS REF CURSOR
RETURN Customers%ROWTYPE;
DECLARE CustRecord Customers%ROWTYPE
BEGIN
OPEN CustCursor;
FETCH CustCursor INTO CustRecord;
CLOSE CustCursor;
END;
在這個例子中,FETCH
用來檢索當前行(自動從第一行開始),放到聲明的變數 CustRecord
中。對於檢索出來的數據不做任何處理。
下一個例子(也使用 Oracle 語法)中,從第一行到最後一行,對檢索出來的數據進行迴圈:
DECLARE TYPE CustCursor IS REF CURSOR
RETURN Customers%ROWTYPE;
DECLARE CustRecord Customers%ROWTYPE
BEGIN
OPEN CustCursor;
LOOP
FETCH CustCursor INTO CustRecord;
EXIT WHEN CustCursor%NOTFOUND;
...
END LOOP;
CLOSE CustCursor;
END;
與前一個例子一樣,這個例子使用 FETCH
檢索當前行,放到一個名為 CustRecord
的變數中。
但不一樣的是,這裡的 FETCH
位於 LOOP
內,因此它反覆執行。
代碼 EXIT WHEN CustCursor%NOTFOUND
使在取不出更多的行時終止處理(退出迴圈)。
這個例子也沒有做實際的處理,實際例子中可用具體的處理代碼替換省略號。
下麵是另一個例子,這次使用 Microsoft SQL Server 語法:
DECLARE @cust_id CHAR(10),
@cust_name CHAR(50),
@cust_address CHAR(50),
@cust_city CHAR(50),
@cust_state CHAR(5),
@cust_zip CHAR(10),
@cust_country CHAR(50),
@cust_contact CHAR(50),
@cust_email CHAR(255)
OPEN CustCursor
FETCH NEXT FROM CustCursor
INTO @cust_id, @cust_name, @cust_address,
@cust_city, @cust_state, @cust_zip,
@cust_country, @cust_contact, @cust_email
...
WHILE @@FETCH_STATUS = 0
BEGIN
FETCH NEXT FROM CustCursor
INTO @cust_id, @cust_name, @cust_address,
@cust_city, @cust_state, @cust_zip,
@cust_country, @cust_contact, @cust_email
...
END
CLOSE CustCursor
在此例中,為每個檢索出的列聲明一個變數,FETCH
語句檢索一行並保存值到這些變數中。
使用 WHILE
迴圈處理每一行,條件 WHILE @@FETCH_STATUS = 0
在取不出更多的行時終止處理(退出迴圈)。
這個例子也不進行具體的處理,實際代碼中,應該用具體的處理代碼替換其中的“...”。
2.3 關閉游標
如前面幾個例子所述,游標在使用完畢時需要關閉。此外,SQL Server 等 DBMS 要求明確釋放游標所占用的資源。
下麵是 DB2、Oracle 和 PostgreSQL 的語法。
CLOSE CustCursor
下麵是 Microsoft SQL Server 的版本。
CLOSE CustCursor
DEALLOCATE CURSOR CustCursor
CLOSE
語句用來關閉游標。一旦游標關閉,如果不再次打開,將不能使用。第二次使用它時不需要再聲明,只需用 OPEN
打開它即可。
三、小結
本文介紹了什麼是游標,為什麼使用游標。
你使用的 DBMS 可能會提供某種形式的游標,以及這裡沒有提及的功能。更詳細的內容請參閱具體的 DBMS 文檔。
原文鏈接:https://www.developerastrid.com/sql/sql-cursor/
(完)