如何創建和使用 SQL 游標

来源:https://www.cnblogs.com/vin-c/archive/2022/05/19/16288637.html
-Advertisement-
Play Games

本文介紹什麼是 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 語句用來定義和命名游標,這裡為 CustCursorSELECT 語句定義一個包含沒有電子郵件地址(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/

(完)


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

-Advertisement-
Play Games
更多相關文章
  • 本文例子參考《STM32單片機開發實例——基於Proteus虛擬模擬與HAL/LL庫》 源代碼:https://github.com/LanLinnet/STM33F103R6 項目要求 單片機將由串口收到的1位元組數據存入Flash ROM的指定地址;按下按鈕BTN,單片機將存儲在Flash ROM ...
  • 為什麼要使用Redwood Redwood是一個全棧web框架,旨在幫助你從副業項目發展到創業。Redwood的特色是一個端到端的開發工作流,它將React、GraphQL、Prisma、TypeScript、Jest和Storybook中最好的部分編織在一起。 RedwoodJS 是集成 、Pri ...
  • Termius是微軟的一款SSH終端工具,它支持多平臺。而且操作界面十分ha好看且簡潔,今天分享給大家❤️ 軟體下載 關註下方公眾號,回覆termius獲取下載地址 軟體功能介紹 Termius Mac破解版是一款非常好用而且漂亮的SSH客戶端,能快速遠程式控制制伺服器,可以定製自己喜歡的主題,支持FT ...
  • 一、概述 Impala 直接針對存儲在 HDFS、HBase或 Amazon Simple Storage Service (S3)中的 Apache Hadoop 數據提供快速的互動式 SQL 查詢。Impala是一個基於Hive、分散式、大規模並行處理(MPP:Massively Paralle ...
  • 軟硬體環境 軟體 版本 操作系統 Redhat 7 及以上版本,麒麟V10 DM 資料庫 DM 8.0 及以上版本 CPU 架構 x86、ARM、龍芯、飛騰等國內外主流 CPU DM-Oracle 環境準備 在DM伺服器中完成Dblink環境準備工作。 配置Oracle oci客戶端 在Oracle ...
  • 導讀: 隨著全球數據量的不斷增長,越來越多的業務需要支撐高併發、高可用、可擴展、以及海量的數據存儲,在這種情況下,適應各種場景的數據存儲技術也不斷的產生和發展。與此同時,各種資料庫之間的同步與轉化的需求也不斷增多,數據集成成為大數據領域的熱門方向,於是SeaTunnel應運而生。SeaTunnel是 ...
  • hive 存儲格式有很多,但常用的一般是 TextFile、ORC、Parquet 格式,在我們單位最多的也是這三種 hive 預設的文件存儲格式是 TextFile。 除 TextFile 外的其他格式的表不能直接從本地文件導入數據,要先導入到 TextFile 格式的表中,再從表中用 inser ...
  • 本文介紹 SQL 所涉及的幾個數據處理特性:約束、索引和觸發器。約束是實施引用完整性的重要部分,索引可改善數據檢索的性能,觸發器可以用來執行運行前後的處理。 一、約束 SQL 已經改進過多個版本,成為非常完善和強大的語言。許多強有力的特性給用戶提供了高級的數據處理技術,如約束。 關聯表和引用完整性已 ...
一周排行
    -Advertisement-
    Play Games
  • 什麼是工廠模式 工廠模式是最常用的設計模式之一,屬於創建型模式。 有點: 解耦,可以把對象的創建和過程分開 減少代碼量,易於維護 什麼時候用? 當一個抽象類有多個實現的時候,需要多次實例化的時候,就要考慮使用工廠模式。 比如:登錄的抽象類ILoginBusiness,它有2個實現,一個用用戶名密碼登 ...
  • 這次iNeuOS升級主要升級圖形渲染引擎和增加豐富的圖元信息,可以很快的方案應用。總共增加41個通用和行業領域的圖元應用,增加2154個圖元信息,現在iNeuOS視圖建模功能模塊總共包括5894個行業圖元信息。現在完全支持製作高保真的工藝流程和大屏展示效果。 ...
  • 效果圖先附上: 首先 這是我是參考 教程:使用 SignalR 2 和 MVC 5 實時聊天 | Microsoft Docs 先附上教程: 在“添加新項 - SignalRChat”中,選擇 InstalledVisual> C#>WebSignalR>,然後選擇 SignalR Hub 類 (v ...
  • 一、前言 項目中之前涉及到胎兒心率圖曲線的繪製,最近項目中還需要添加心電曲線和血樣曲線的繪製功能。今天就來分享一下心電曲線的繪製方式; 二、正文 1、胎兒心率曲線的繪製是通過DrawingVisual來實現的,這裡的心電曲線我也是採用差不多相同的方式來實現的,只是兩者曲線的數據有所區別。心電圖的數據 ...
  • 安裝 Redis # 首先安裝依賴gcc, 後面需要使用make編譯redis yum install gcc -y # 進入 /usr/local/src 目錄, 把源碼下載到這裡 cd /usr/local/src # 下載 redis 7.0.2 的源碼,github被牆,可以使用國內的地址 ...
  • Redis 的定義? 百度百科: Redis(Remote Dictionary Server ),即遠程字典服務,是一個開源的使用ANSI C語言編寫、支持網路、可基於記憶體亦可持久化的日誌型、Key-Value資料庫,並提供多種語言的API。 中文官網: Redis是一個開源(BSD許可),記憶體存 ...
  • 事情的起因是收到了一位網友的請求,他的java課設需要設計實現迷宮相關的程式——如標題概括。 我這邊不方便透露相關信息,就只把任務要求寫出來。 演示視頻指路👉: 基於JavaFX圖形界面的迷宮程式演示_嗶哩嗶哩_bilibili 完整代碼鏈接🔎: 網盤:https://pan.baidu.com ...
  • Python中的字典 Python中的字典是另一種可變容器模型,且可存儲任意類型對象。鍵值使用冒號分割,你可以看成是一串json。 常用方法 獲取字典中的值 dict[key] 如果key不存在會報錯,建議使用dict.get(key),不存在返回None 修改和新建字典值 dict[key]=va ...
  • 迎面走來了你的面試官,身穿格子衫,挺著啤酒肚,髮際線嚴重後移的中年男子。 手拿泡著枸杞的保溫杯,胳膊夾著MacBook,MacBook上還貼著公司標語:“加班使我快樂”。 面試官: 看你簡歷上用過MySQL,問你幾個簡單的問題吧。什麼是聚簇索引和非聚簇索引? 這個問題難不住我啊。來之前我看一下一燈M ...
  • tunm二進位協議在python上的實現 tunm是一種對標JSON的二進位協議, 支持JSON的所有類型的動態組合 支持的數據類型 基本支持的類型 "u8", "i8", "u16", "i16", "u32", "i32", "u64", "i64", "varint", "float", "s ...