Total Pageviews

Saturday, 3 May 2014

Understanding Ref Cursors with Examples part 1



Today we are going to discuss about system reference cursor or ref cursor. I have been working in IT industry for last 7 years and I have seen many developers who avoid using this very powerful tool as their understanding of ref cursor is not as clear as simple cursor. So in this edition we will try to explore various aspects of ref cursor and how can we use those features in daily programming which will make our life much easier.
First question which I would like to answer before going in detailed discussion is what is ref cursor.
System reference cursors or Ref cursors are pointers to the Executed Query in an area called  process global area (PGA).

Second question is what is cursor variable? Cursor variable is a variable that points to the reference of a Cursor. A cursor variable (especially weak cursor variable, discussed later) can be opened for any number of queries. User can use a cursor variable multiple times within a same execution unit.

Some of the biggest benefit of cursor variable is written below
*       Programmer can use the ref cursor as a parameter to other pl/sql units.
*      Programmer can assign the pointer of one cursor to another cursor variable. Like other variables cursor variables can be assigned values using an assignment operator (See the last example).
But before using ref cursor in program all programmers should keep in mind the below points
*      A ref cursor involves additional database round tip. While the ref cursor is returned to the      client, the actual data is not returned until client opens the ref cursor and try to read it.
*      Secondly, you cannot update the database by ref cursor.
*       Lastly, Ref cursor is not backward scrollable. Data can be fetched only in forward directions.

Types of Ref Cursor:-
The Syntax for creating a system Referenced Cursor or Ref Cursor is as follows
TYPE  cursor_name IS REF CURSOR [ RETURN return_type ].
The portion which exists within the brace is an optional part of ref cursor declaration.

Based on the existence of return clause there are two types of Ref Cursor

1>      Strong Ref Cursor :- This type of ref cursors have always a return type.
For example
TYPE v_emp IS REF CURSOR RETURN emp%ROWTYPE;
v_emp_rec   v_emp;

This type of Ref Cursor can only fetch emp%rowtype data structure. This cursor will not work on any other data structure. The advantage of a strong type is that the compiler can determine whether or not the developer has properly matched up the cursor variable FETCH statements with it’s cursor objects query lists.
For Example:-
DECLARE
  TYPE v_emp IS REF CURSOR RETURN emp%ROWTYPE;
  v_emp_ref v_emp;
  emp_rec   emp%ROWTYPE;
BEGIN
  OPEN v_emp_ref FOR SELECT * FROM emp;
  LOOP
    FETCH v_emp_ref INTO emp_rec;
    EXIT WHEN v_emp_ref%NOTFOUND;
    DBMS_OUTPUT.PUT_LINE(emp_rec.empno||'--'||emp_rec.ename||'--'||emp_rec.job||'--'||emp_rec.mgr);
  END LOOP;
END;

2>     Weak Ref Cursor :- This type of ref cursors does not have return type.
For example
TYPE v_emp_ref IS REF CURSOR;
OR
v_emp_ref  SYS_REFCURSOR; (SYS_REFCURSOR IS a predefined weak REF CURSOR type).

This type of Ref Cursor can be associated with any query. As it does not return any kind of particular data structure it can be associated with different queries which have different kind of result set. It is more flexible than the Strong Ref Cursor.

For example
DECLARE
  v_emp_ref SYS_REFCURSOR;
  emp_rec   emp%ROWTYPE;
  dept_rec  dept%ROWTYPE;
BEGIN
  DBMS_OUTPUT.PUT_LINE('----------------------------------------');
  DBMS_OUTPUT.PUT_LINE('----- Fetching Data from emp table-----');
  DBMS_OUTPUT.PUT_LINE('----------------------------------------');
  OPEN v_emp_ref FOR SELECT * FROM emp;
  LOOP
    FETCH v_emp_ref INTO emp_rec;
    EXIT WHEN v_emp_ref%NOTFOUND;
    DBMS_OUTPUT.PUT_LINE(emp_rec.empno||'--'||emp_rec.ename||'--'||emp_rec.job||'--'||emp_rec.mgr);
  END LOOP;
  CLOSE v_emp_ref;   -- Good practice code
  --
  DBMS_OUTPUT.PUT_LINE('----------------------------------------');
  DBMS_OUTPUT.PUT_LINE('----- Fetching Data from Dept table-----');
  DBMS_OUTPUT.PUT_LINE('----------------------------------------');
  --
  OPEN v_emp_ref FOR SELECT * FROM dept;
  LOOP
    FETCH v_emp_ref INTO dept_rec;
    EXIT WHEN v_emp_ref%NOTFOUND;
    DBMS_OUTPUT.PUT_LINE(dept_rec.deptno||'---'||dept_rec.dname||'--'||dept_rec.loc);
  END LOOP;
END;
And the result we got is
----------------------------------------
----- Fetching Data from emp table-----
----------------------------------------
7400--SMITH--CLERK--7902
7369--SMITH--CLERK--7902
7499--ALLEN--SALESMAN--7698
7566--JONES--MANAGER--7839
7654--MARTIN--SALESMAN--7698
7698--BLAKE--MANAGER--7839
7782--CLARK--MANAGER--7839
7788--SCOTT--ANALYST--7566
7839--KING--PRESIDENT--
7844--TURNER--SALESMAN--7698
7876--ADAMS--CLERK--7788
7900--JAMES--CLERK--7698
7902--FORD--ANALYST--7566
7934--MILLER--CLERK—7782

----------------------------------------
----- Fetching Data from Dept table-----
----------------------------------------
25---MANUFACTURING--KOLKATA
10---ACCOUNTING--NEW YORK
20---RESEARCH--DALLAS
30---SALES--CHICAGO
40---OPERATIONS--BOSTON

Please observe that the single ref cursor is used to query two different tables in two different queries. Secondly, when you are associating a new result set with a cursor variable that was previously used in an OPEN FOR statement and you did not explicitly close the cursor variable, then the underlying cursor remain open. It is not mandatory but you should always close the cursor variable before reusing them with another result set. Lastly, declaration of cursor variable does not create a cursor object. Programmer must use the OPEN FOR syntax to create a new cursor object and assign it to cursor variable.

Similarities with Static Cursor:-

While using Ref Cursor programmer can always use the basic four attributes with exactly the same syntax as for a static cursor. They are as follows
1>     REF_CUR%ISOPEN
2>     REF_CUR%FOUND
3>     REF_CUR%NOTFOUND
4>     REF_CUR%ROWCOUNT
With the following example the points will be clarified. We will modify our previous example a little bit.

SET SERVEROUTPUT ON SIZE 20000;
DECLARE
  TYPE v_emp IS REF CURSOR RETURN emp%ROWTYPE;
  v_emp_ref v_emp;
  emp_rec   emp%ROWTYPE;
BEGIN
  OPEN v_emp_ref FOR SELECT * FROM emp;
  ------
  IF v_emp_ref%ISOPEN THEN
    DBMS_OUTPUT.PUT_LINE('v_emp_ref cursor OPEN');
  END IF;
  ------
  DBMS_OUTPUT.PUT_LINE('Rowcount--->'||v_emp_ref%ROWCOUNT);
  ------
  LOOP
    FETCH v_emp_ref INTO emp_rec;
    EXIT WHEN v_emp_ref%NOTFOUND;
      DBMS_OUTPUT.PUT_LINE(emp_rec.empno||'--'||emp_rec.ename||'--'||emp_rec.job||'--'||emp_rec.mgr);
    DBMS_OUTPUT.PUT_LINE('Rowcount--->'||v_emp_ref%ROWCOUNT);
    IF v_emp_ref%FOUND THEN
      DBMS_OUTPUT.PUT_LINE('v_emp_ref cursor found');
    END IF;
  END LOOP;
  DBMS_OUTPUT.PUT_LINE('Final Rowcount--->'||v_emp_ref%ROWCOUNT);
END;
RESULT:-

anonymous block completed
v_emp_ref cursor OPEN
Rowcount--->0
7400--SMITH--CLERK--7902
Rowcount--->1
v_emp_ref cursor found
7369--SMITH--CLERK--7902
Rowcount--->2
v_emp_ref cursor found
7499--ALLEN--SALESMAN--7698
Rowcount--->3
v_emp_ref cursor found
7566--JONES--MANAGER--7839
Rowcount--->4
v_emp_ref cursor found
7654--MARTIN--SALESMAN--7698
Rowcount--->5
v_emp_ref cursor found
7698--BLAKE--MANAGER--7839
Rowcount--->6
v_emp_ref cursor found
7782--CLARK--MANAGER--7839
Rowcount--->7
v_emp_ref cursor found
7788--SCOTT--ANALYST--7566
Rowcount--->8
v_emp_ref cursor found
7839--KING--PRESIDENT--
Rowcount--->9
v_emp_ref cursor found
7844--TURNER--SALESMAN--7698
Rowcount--->10
v_emp_ref cursor found
7876--ADAMS--CLERK--7788
Rowcount--->11
v_emp_ref cursor found
7900--JAMES--CLERK--7698
Rowcount--->12
v_emp_ref cursor found
7902--FORD--ANALYST--7566
Rowcount--->13
v_emp_ref cursor found
7934--MILLER--CLERK--7782
Rowcount--->14
v_emp_ref cursor found
Final Rowcount--->14

Please observe one thing that at the very beginning of the execution the ROWCOUNT is 0 because at that time no row were fetched from the underlying table.

ROWTYPE_MISMATCH EXCEPTION:-

Some time programmer who is accustomed to use Strong Ref Cursor may get it. It is an exception which tell programmer that the data structure of query is different than the return type of the Ref Cursor itself.

DECLARE
  --v_emp_ref SYS_REFCURSOR;
  TYPE v_emp IS REF CURSOR RETURN emp%ROWTYPE;
  v_emp_ref v_emp;
  emp_rec   emp%ROWTYPE;
  PROCEDURE Return_Dept_Row(v_emp IN OUT SYS_REFCURSOR)
  IS
  BEGIN
    OPEN v_emp FOR SELECT * FROM dept;
  END Return_Dept_Row;
BEGIN
  Return_Dept_Row(v_emp_ref);
  LOOP
    FETCH v_emp_ref INTO emp_rec;
    EXIT WHEN v_emp_ref%NOTFOUND;
    DBMS_OUTPUT.PUT_LINE(emp_rec.empno||'--'||emp_rec.ename||'--'||emp_rec.job||'--'||emp_rec.mgr);
  END LOOP;
EXCEPTION
  WHEN ROWTYPE_MISMATCH THEN
    DBMS_OUTPUT.PUT_LINE('Within ROWTYPE_MISMATCH: Reason for Error-->'||SQLERRM);
  WHEN OTHERS THEN
    DBMS_OUTPUT.PUT_LINE('Within Others: Reason for Error-->'||SQLERRM);
END;

Assigning one Ref Cursor to another Ref Cursor variable :-

As ref cursor is just another kind of variable programmer can assign the reference of one ref cursor variable to another just using an assignment operator.

For example

DECLARE
  v_emp_ref     SYS_REFCURSOR;
  v_emp_sec_ref SYS_REFCURSOR;
  emp_rec       emp%ROWTYPE;
  emp_rec_sec   emp%ROWTYPE;
BEGIN
  OPEN v_emp_ref FOR SELECT * FROM emp;
  LOOP
    FETCH v_emp_ref INTO emp_rec;
    EXIT WHEN v_emp_ref%NOTFOUND;
    DBMS_OUTPUT.PUT_LINE(emp_rec.empno||'--'||emp_rec.ename||'--'||emp_rec.job||'--'||emp_rec.mgr);
    --
    v_emp_sec_ref := v_emp_ref;
    FETCH v_emp_ref INTO emp_rec_sec;
     DBMS_OUTPUT.PUT_LINE(emp_rec_sec.empno||'--'||emp_rec_sec.ename||'--'||emp_rec_sec.job||'--'||emp_rec_sec.mgr);
    CLOSE v_emp_sec_ref;
  END LOOP;
  DBMS_OUTPUT.PUT_LINE('Final Rowcount--->'||v_emp_ref%ROWCOUNT);
END;
OutPut is:-

ORA-01001: invalid cursor
ORA-06512: at line 9
01001. 00000 -  "invalid cursor"
*Cause:   
*Action:
7400--SMITH--CLERK--7902
7369--SMITH--CLERK—7902

Please observe that even v_emp_ref and v_emp_ref_sec is two different ref Cursor variable and both ref Cursor variable has fetched two different set of data from emp table because the pointer to the current record stays with the cursor object. It does not stays with any ref cursor variable.

During execution when we have close v_emp_ref_sec cursor variable it also closed the ref cursor v_emp_ref.