分類: PL/SQL開發 2013-12-04 15:15 685人閱讀
一 介紹java
像遊標cursor同樣,遊標變量ref cursor指向指定查詢結果集當前行。遊標變量顯得更加靈活由於其聲明並不綁定指定查詢。sql
其主要運用於PLSQL函數或存儲過程以及其餘編程語言java等程序之間做爲參數傳遞。編程
不像遊標的一點,遊標變量沒有參數。數組
遊標變量具備如下屬性:oracle
(%FOUND, %NOTFOUND, %ISOPEN, and %ROWCOUNT)編程語言
二 用法介紹:ide
一、聲明格式:函數
DECLARE oop
TYPE DeptCurTyp IS REF CURSOR RETURN departments%ROWTYPE;post
二、遊標變量又分爲強類型strong(with a return type)和弱類型(with no return type):
DECLARE
TYPE empcurtyp IS REF CURSOR RETURN employees%ROWTYPE; -- 強類型
TYPE genericcurtyp IS REF CURSOR; -- 弱類型
cursor1 empcurtyp;
cursor2 genericcurtyp;
my_cursor SYS_REFCURSOR; -- 使用預約義遊標變量sys_refcursor
TYPE deptcurtyp IS REF CURSOR RETURN departments%ROWTYPE;
dept_cv deptcurtyp; -- 聲明遊標變量
或是返回record類型:
DECLARE
TYPE EmpRecTyp IS RECORD (
employee_id NUMBER,
last_name VARCHAR2(25),
salary NUMBER(8,2));
TYPE EmpCurTyp IS REF CURSOR RETURN EmpRecTyp;
emp_cv EmpCurTyp; -- declare cursor variable
三、使用遊標變量做爲參數傳遞:
- DECLARE
- TYPE empcurtyp IS REF CURSOR RETURN employees%ROWTYPE;
- emp empcurtyp;
- PROCEDURE process_emp_cv (emp_cv IN empcurtyp) IS
- person employees%ROWTYPE;
- BEGIN
- DBMS_OUTPUT.PUT_LINE('-----');
- DBMS_OUTPUT.PUT_LINE('Here are the names from the result set:');
- LOOP
- FETCH emp_cv INTO person;
- EXIT WHEN emp_cv%NOTFOUND;
- DBMS_OUTPUT.PUT_LINE('Name = ' || person.first_name ||
- ' ' || person.last_name);
- END LOOP;
- END;
- BEGIN
- OPEN emp FOR SELECT * FROM employees WHERE ROWNUM < 11;
- process_emp_cv(emp);
- CLOSE emp;
- OPEN emp FOR SELECT * FROM employees WHERE last_name LIKE 'R%';
- process_emp_cv(emp);
- CLOSE emp;
- END;
- /
四、使用遊標熟悉檢查遊標變量是否打開
- DECLARE
- TYPE empcurtyp IS REF CURSOR RETURN employees%ROWTYPE;
- emp_cv empcurtyp;
- BEGIN
- IF NOT emp_cv%ISOPEN THEN
- OPEN emp_cv FOR SELECT * FROM employees;
- END IF;
- CLOSE emp_cv;
- END;
- /
五、在包package中聲明遊標變量:
- CREATE PACKAGE emp_data AS
- TYPE empcurtyp IS REF CURSOR RETURN employees%ROWTYPE;
- PROCEDURE open_emp_cv (emp_cv IN OUT empcurtyp);
- END emp_data;
- /
- CREATE PACKAGE BODY emp_data AS
- PROCEDURE open_emp_cv (emp_cv IN OUT EmpCurTyp) IS
- BEGIN
- OPEN emp_cv FOR SELECT * FROM employees;
- END open_emp_cv;
- END emp_data;
- /
六、提取遊標變量到集合類型collection:
- DECLARE
- TYPE empcurtyp IS REF CURSOR;
- TYPE namelist IS TABLE OF employees.last_name%TYPE;
- TYPE sallist IS TABLE OF employees.salary%TYPE;
- emp_cv empcurtyp;
- names namelist;
- sals sallist;
- BEGIN
- OPEN emp_cv FOR SELECT last_name, salary FROM employees
- WHERE job_id = 'SA_REP';
- FETCH emp_cv BULK COLLECT INTO names, sals;
- CLOSE emp_cv;
- FOR i IN names.FIRST .. names.LAST
- LOOP
- DBMS_OUTPUT.PUT_LINE('Name = ' || names(i) || ', salary = ' || sals(i));
- END LOOP;
- END;
- /
三 遊標變量的使用限制:
一、不能再包說明中聲明遊標變量;
二、不能用「=」運算符比較遊標變量相等性、不等性及是否爲空;
三、不能存儲於表列中;
四、不能將遊標變量存在於關聯數組、嵌套表或數組;
五、遊標和遊標變量以前是不可互操做的!
--------------------------------------------------------------------------------------
附:
------------------------
一、強類型遊標:
- CREATE OR REPLACE PACKAGE strongly_typed IS
-
- TYPE return_cur IS REF CURSOR RETURN all_tables%ROWTYPE;
- PROCEDURE child(p_return_rec OUT return_cur);
- PROCEDURE parent(p_NumRecs PLS_INTEGER);
-
- END strongly_typed;
- /
- CREATE OR REPLACE PACKAGE BODY strongly_typed IS
- PROCEDURE child(p_return_rec OUT return_cur) IS
-
- BEGIN
- OPEN p_return_rec FOR
- SELECT * FROM all_tables;
- END child;
- PROCEDURE parent (p_NumRecs PLS_INTEGER) IS
- p_retcur return_cur;
- at_rec all_tables%ROWTYPE;
- BEGIN
- child(p_retcur);
-
- FOR i IN 1 .. p_NumRecs
- LOOP
- FETCH p_retcur
- INTO at_rec;
-
- dbms_output.put_line(at_rec.table_name ||
- ' - ' || at_rec.tablespace_name ||
- ' - ' || TO_CHAR(at_rec.initial_extent) ||
- ' - ' || TO_CHAR(at_rec.next_extent));
- END LOOP;
- END parent;
- END strongly_typed;
- /
- set serveroutput on
-
- exec strongly_typed.parent(1);
- exec strongly_typed.parent(8);
二、弱類型遊標:
- CREATE OR REPLACE PROCEDURE child (
- p_NumRecs IN PLS_INTEGER,
- p_return_cur OUT SYS_REFCURSOR)
- IS
-
- BEGIN
- OPEN p_return_cur FOR
- 'SELECT * FROM all_tables WHERE rownum <= ' || p_NumRecs ;
- END child;
- /
-
- CREATE OR REPLACE PROCEDURE parent (pNumRecs VARCHAR2) IS
- p_retcur SYS_REFCURSOR;
- at_rec all_tables%ROWTYPE;
- BEGIN
- child(pNumRecs, p_retcur);
-
- FOR i IN 1 .. pNumRecs
- LOOP
- FETCH p_retcur
- INTO at_rec;
-
- dbms_output.put_line(at_rec.table_name ||
- ' - ' || at_rec.tablespace_name ||
- ' - ' || TO_CHAR(at_rec.initial_extent) ||
- ' - ' || TO_CHAR(at_rec.next_extent));
- END LOOP;
- END parent;
- /
-
- set serveroutput on
-
- exec parent(1);
- exec parent(17);
三、預約義遊標變量:
- CREATE TABLE employees (
- empid NUMBER(5),
- empname VARCHAR2(30));
-
- INSERT INTO employees (empid, empname) VALUES (1, 'Dan Morgan');
- INSERT INTO employees (empid, empname) VALUES (2, 'Hans Forbrich');
- INSERT INTO employees (empid, empname) VALUES (3, 'Caleb Small');
- COMMIT;
- CREATE OR REPLACE PROCEDURE pass_ref_cur(p_cursor SYS_REFCURSOR) IS
-
- TYPE array_t IS TABLE OF VARCHAR2(4000)
- INDEX BY BINARY_INTEGER;
-
- rec_array array_t;
-
- BEGIN
- FETCH p_cursor BULK COLLECT INTO rec_array;
-
- FOR i IN rec_array.FIRST .. rec_array.LAST
- LOOP
- dbms_output.put_line(rec_array(i));
- END LOOP;
- END pass_ref_cur;
- /
- set serveroutput on
-
- DECLARE
- rec_array SYS_REFCURSOR;
- BEGIN
- OPEN rec_array FOR
- 'SELECT empname FROM employees';
-
- pass_ref_cur(rec_array);
- CLOSE rec_array;
- END;
- /