" 1、類型 " "1.1、屬性類型" "1.2、記錄類型" " 2、變數 " "2.1、變數類型" "2.2、變數定義" "2.3、變數賦值" " 3、結構 " "3.1、順序結構" "3.2、選擇結構" "3.3、迴圈結構" " 4、總結 " 1、類型 在 "《.Net程式員學用Oracle系列 ...
1、類型
在《.Net程式員學用Oracle系列(5):三大數據類型》一文中詳細地講述了 Oracle 的基本數據類型,文中還提到,除基本數據類型之外,Oracle 還在語法上支持一些非固有數值類型。
事實上,Oracle 在語法上支持的數據類型遠不止於此,Oracle 還支持一些複雜而強大的數據類型。如屬性類型%TYPE
和%ROWTYPE
,記錄類型 RECORD,集合類型 VARRAY 和 TABLE 等。本節將會介紹實用的屬性類型和靈活的記錄類型。
1.1、屬性類型
屬性類型是一種可以直接引用資料庫中列的數據類型來描述變數類型的類型。Oracle 提供了兩種屬性類型,分別是%TYPE
和%ROWTYPE
,下文將逐一說明。
%TYPE:該屬性允許在聲明中引用資料庫中的列或先前聲明的變數的數據類型,而不是硬編碼類型名稱。在聲明常量、變數和參數時,都可以使用%TYPE
屬性作為數據類型說明符。如果引用的類型被更改,則聲明也將自動更新,這點有利於後期代碼維護。
示例:
DECLARE
v_staff_name t_staff.staff_name%TYPE;
BEGIN
SELECT t.staff_name INTO v_staff_name FROM demo.t_staff t WHERE t.staff_id=5;
DBMS_OUTPUT.PUT_LINE(v_staff_name);
END;
%ROWTYPE:該屬性可以表示資料庫中表或游標的行的記錄類型。使用%ROWTYPE
聲明的變數可以存儲從表中選擇或從游標或游標變數獲取的整行數據,且變數記錄中的欄位和每行中的相應列具有相同的名稱和數據類型。
示例:
DECLARE
v_staff t_staff%ROWTYPE;
BEGIN
SELECT t.* INTO v_staff FROM demo.t_staff t WHERE t.staff_id=5;
DBMS_OUTPUT.PUT_LINE(v_staff.staff_name);
END;
想要更多的瞭解屬性類型可參考:《Database PL/SQL User's Guide and Reference: %TYPE Attribute》和《Database PL/SQL User's Guide and Reference: %ROWTYPE Attribute》。
1.2、記錄類型
記錄類型是由單行多列標量構成的複合結構。可以看做是一種用戶自定義的數據類型,提供了將一個或多個標量封裝成一個對象進行操作的能力。在使用記錄數據類型的變數時,需要在聲明部分先定義記錄的成員變數,然後在執行部分引用該記錄變數本身或其中的成員。但不可以對記錄做整體性的比較運算,如判斷記錄類型的變數是否為 NULL。
示例:
DECLARE
TYPE staff_type IS RECORD(
staff_name VARCHAR2(50),
gender VARCHAR2(2)
);
v_staff staff_type;
BEGIN
SELECT t.staff_name,DECODE(t.gender,1,'男',0,'女','兩性') INTO v_staff.staff_name,v_staff.gender
FROM demo.t_staff t WHERE t.staff_id=5;
DBMS_OUTPUT.PUT_LINE(v_staff.staff_name||'|'||v_staff.gender);
END;
記錄類型和%ROWTYPE
屬性在用途上比較相似,區別在於前者是自定義結構,而後者為表結構,前者比較靈活,而後者比較方便。
2、變數
一般電腦編程語言中都有變數的概念,PL/SQL 也不例外,變數用於存儲計算結果和表示可變狀態,本節將著重介紹在 PL/SQL 中變數如何定義及賦值。另外,PL/SQL 中也有常量,只是極少有人使用,有興趣的讀者可以參考《Oracle Database PL/SQL Language Reference: Constant》。
2.1、變數類型
在 PL/SQL 中定義變數的可選類型非常多,包括 Oracle 中常見的三大類基本數據類型,以及 Oracle 在語法上支持的諸多非固有數據類型。如整形(INT/INTEGER)、布爾類型(BOOLEAN)等 Oracle 本身並不支持,卻在 PL/SQL 中可用的數據類型。
2.2、變數定義
在 PL/SQL 中定義變數與 C# 中定義變數本質上並無區別,不同的是 PL/SQL 中的變數得集中定義,變數定義區域得用DECLARE
關鍵字開頭,且每行只能定義一個變數。如果是 SQL*Plus 環境則必須用VAR[IABLE]
開頭。
語法:
variable_name datatype [[NOT NULL] {:= | DEFAULT} expression];
如果使用了 NOT NULL 則必須給變數賦初始值。另外,在命名變數的時候還需要遵守以下命名規則:
- 1、變數名必須以字母開頭。
- 2、變數名長度不能超過 30 個字元。
- 3、變數名中不能含有空格。
- 4、同一語句塊內變數名不能重覆。
- 5、變數名不能與查詢中的列名相同。
2.3、變數賦值
給 PL/SQL 變數賦值的寫法與給 C# 欄位賦值寫法基本一樣,既可以在定義變數時就賦給它一個初始值,也可以在使用之前給它賦值,如果不賦值也會有預設值。唯一的區別就是,C# 中類型不同預設值也不同,而 PL/SQL 中所有類型的預設值都一樣,都是 NULL。
在 PL/SQL 中還有一點比較怪的就是,可能所有編程語言的賦值操作符都是=
,而 PL/SQL 中的賦值操作符卻偏偏是:=
。而且 Oracle 相關的 API 中參數寫法也與其它大多數資料庫不同。
示例一(普通 PL/SQL 環境):
DECLARE
v1 NUMBER;
v2 NUMBER(5,2);
v3 NUMBER := 50.20;
v4 NUMBER(4) := 1998;
v5 VARCHAR2(4) DEFAULT 'A';
v6 DATE NOT NULL := fn_now;
BEGIN
v1 := 100;
v2 := 99.99;
v5 := 'A5';
v6 := SYSDATE;
DBMS_OUTPUT.PUT_LINE(v1||'|'||v2||'|'||v3||'|'||v4||'|'||v5||'|'||v6);
END;
示例二(SQL*Plus 環境):
VARIABLE v1 NUMBER
BEGIN
:v1 := 12;
DBMS_OUTPUT.PUT_LINE(:v1);
END;
/
3、結構
和普通編程語言一樣,PL/SQL 中也有常見的三大控制結構以及順序控制語句——GOTO。本節將重點講述被廣泛接受的三大控制結構,至於不受待見的 GOTO 語句,有興趣的讀者可以參考《Oracle Database PL/SQL User's Guide and Reference: Using the GOTO Statement》。
3.1、順序結構
順序結構是面向過程編程中最基本、最簡單、最常用的程式控制結構。順序結構用於表示若幹個依次執行的處理步驟,表現形式就是線性結構,一個方向走下去、不拐彎。使用時只要按照解決問題的順序寫出相應的語句就行,它的執行順序是自上而下、依次執行。
3.2、選擇結構
PL/SQL 中提供了兩種選擇結構,分別是IF
結構和CASE
結構。其中IF
結構有三種變體,CASE
結構有兩種變體。下文將逐一講述各個選擇語句:
IF 結構變體一:
語法:
IF condition THEN
{...statements to execute when condition is TRUE...}
END IF;
示例:
BEGIN
IF 1>0 THEN
DBMS_OUTPUT.PUT_LINE('executed');
END IF;
END;
IF 結構變體二:
語法:
IF condition THEN
{...statements to execute when condition is TRUE...}
ELSE
{...statements to execute when condition is FALSE...}
END IF;
示例:
BEGIN
IF 1>2 THEN
DBMS_OUTPUT.PUT_LINE('The result is true');
ELSE
DBMS_OUTPUT.PUT_LINE('The result is false');
END IF;
END;
IF 結構變體三:
語法:
IF condition1 THEN
{...statements to execute when condition1 is TRUE...}
ELSIF condition2 THEN
{...statements to execute when condition2 is TRUE...}
[ELSE
{...statements to execute when both condition1 and condition2 are FALSE...}]
END IF;
示例:
BEGIN
IF 1>2 THEN
DBMS_OUTPUT.PUT_LINE('1>2 branch');
ELSIF 1<2 THEN
DBMS_OUTPUT.PUT_LINE('1<2 branch');
ELSE
DBMS_OUTPUT.PUT_LINE('1=2 branch');
END IF;
END;
註意:IF 結構變體三中有個巨坑,就是 IF 和 ELSE 之間的分支寫法,不是 ELSE IF 也不是 ELSEIF 而是 ELSIF。儘管你寫成 ELSE IF 編輯器也有智能提示,但當你執行的時候就會報ORA-06550
的錯,而且這個錯誤的描述正常人基本沒可能看懂。
CASE 結構變體一
示例:
DECLARE
v_grade VARCHAR2(1);
BEGIN
v_grade:='B';
CASE v_grade
WHEN 'A' THEN DBMS_OUTPUT.PUT_LINE('甲');
WHEN 'B' THEN DBMS_OUTPUT.PUT_LINE('乙');
WHEN 'C' THEN DBMS_OUTPUT.PUT_LINE('丙');
ELSE DBMS_OUTPUT.PUT_LINE('丁');
END CASE;
END;
CASE 結構變體二
示例:
DECLARE
v_score NUMBER(3);
BEGIN
v_score:=78;
CASE
WHEN v_score>=80 THEN DBMS_OUTPUT.PUT_LINE('優');
WHEN v_score>=70 THEN DBMS_OUTPUT.PUT_LINE('良');
WHEN v_score>=60 THEN DBMS_OUTPUT.PUT_LINE('中');
ELSE DBMS_OUTPUT.PUT_LINE('差');
END CASE;
END;
與多分支的 IF 語句相比,CASE 語句更可讀、更高效,所以當程式分支較多時,應儘可能的使用 CASE 而不是 IF。CASE 語句的 ELSE 子句是可選的。但如果省略 ELSE 字句,PL/SQL 將為 CASE 語句添加以下隱式的 ELSE 子句:
ELSE RAISE CASE_NOT_FOUND;
換句話說,如果你省略了 ELSE 子句,且 CASE 語句與 WHEN 子句不匹配,PL/SQL 就會引發預定義的異常CASE_NOT_FOUND
。
3.3、迴圈結構
PL/SQL 中提供了三種迴圈結構,分別是LOOP
、WHILE LOOP
和FOR LOOP
。另外 PL/SQL 還提供了EXIT
語句用於退出當前迴圈。下文將逐一講述各個迴圈語句:
LOOP 迴圈
語法:
LOOP
{...statements...}
EXIT [ WHEN boolean_condition ];
END LOOP;
示例一:
DECLARE
v_counter BINARY_INTEGER := 0;
BEGIN
LOOP
v_counter := v_counter + 1;
DBMS_OUTPUT.PUT_LINE(v_counter); -- 輸出結果:1、2、3、4、5、6、7、8、9
IF v_counter >= 9 THEN
EXIT;
END IF;
-- 上面的 IF 語句塊還可以由“EXIT WHEN v_counter >= 9;”代替
END LOOP;
END;
示例二(嵌套迴圈):
DECLARE
i BINARY_INTEGER := 0;
j BINARY_INTEGER := 0;
BEGIN
LOOP
i := i + 1;
j := 0;
LOOP
j := j + 1;
DBMS_OUTPUT.PUT_LINE('i*j=('||i||'*'||j||')='||i*j);
EXIT WHEN j >= 3;
END LOOP;
EXIT WHEN i >= 4;
END LOOP;
END;
示例三(標記迴圈):
DECLARE
i BINARY_INTEGER := 0;
j BINARY_INTEGER := 0;
BEGIN
<<outer_loop>>
LOOP
i := i + 1;
j := 0;
<<inner_loop>>
LOOP
j := j + 1;
DBMS_OUTPUT.PUT_LINE('i*j=('||i||'*'||j||')='||i*j);
EXIT inner_loop WHEN j >= 3;
EXIT outer_loop WHEN i >= 4;
END LOOP inner_loop;
END LOOP outer_loop;
END;
WHILE LOOP 迴圈
語法:
WHILE condition
LOOP
{...statements...}
END LOOP;
示例:
DECLARE
v_score NUMBER(3) := 0;
BEGIN
WHILE v_score < 60 LOOP
v_score := v_score + 10;
DBMS_OUTPUT.PUT_LINE(v_score); -- 輸出結果:10、20、30、40、50、60
END LOOP;
DBMS_OUTPUT.PUT_LINE('over');
END;
FOR LOOP 迴圈
語法:
FOR loop_counter IN [REVERSE] lowest_number..highest_number
LOOP
{...statements...}
END LOOP;
示例一(正向迴圈):
BEGIN
FOR i IN 3..7 LOOP
DBMS_OUTPUT.PUT_LINE(i); -- 輸出結果:3、4、5、6、7
END LOOP;
END;
示例二(反向迴圈):
BEGIN
FOR i IN REVERSE 3..7 LOOP
DBMS_OUTPUT.PUT_LINE(i); -- 輸出結果:7、6、5、4、3
END LOOP;
END;
註意:FOR LOOP 迴圈中的計數器(變數)可以被讀取,但不能被修改。另外,在 LOOP 迴圈的示例中用到的 EXIT 和迴圈標記,同樣可用於 WHILE LOOP 迴圈和 FOR LOOP 迴圈中。
4、總結
本文主要講述了 PL/SQL 中的變數和結構兩個基本的編程元素,以及屬性類型和記錄類型。本人覺得對於常年使用 Oracle 的朋友來說,這幾個知識點應該是必須要掌握的。
本文鏈接:http://www.cnblogs.com/hanzongze/p/Oracle-plsql-1.html
版權聲明:本文為博客園博主 韓宗澤 原創,作者保留署名權!歡迎通過轉載、演繹或其它傳播方式來使用本文,但必須在明顯位置給出作者署名和本文鏈接!本人初寫博客,水平有限,若有不當之處,敬請批評指正,謝謝!