Why use explicit cursors instead of regular loops?
The code you posted is using a cursor. It is using an implicit cursor loop.
There are cases where using an explicit cursor loop (i.e. declaring a CURSOR variable in the declaration section) produces either cleaner code or better performance
- If you have more complex queries that you can't refactor out into views, it can make the code easier to read if your loop iterates over
student_cursor
rather than including a 30 line SQL statement that embeds a bunch of logic. For example, if you were printing out all the students that were cleared to graduate and that involved joining to tables that had their academic records, the requirements of their degree program, tables with information on academic holds, tables with information about overdue library books, tables with information about unpaid fees, administrative overrides, etc. it would probably make sense to refactor the code so that this query wasn't stuck in the middle of code that is concerned with presenting the list to a user. That might involve creating a view that would encapsulate all this logic. Or it might involve creating an explicit cursor that was declared either as part of the current PL/SQL block or in some higher-level PL/SQL block (i.e. a cursor declared in a package) so that it is reusable. Or it might involve doing something else for encapsulation and reusability (say, creating a pipelined table function instead). - If you want to make use of bulk operations in PL/SQL, you generally want to use explicit cursors. Here is a StackOverflow thread that discusses the performance differences between explicit and implicit cursors. If all you are doing is calling
htp.prn
, doing aBULK COLLECT
probably doesn't buy you anything. In other cases, though, it can result in substantial performance improvements.
A cursor can be explicit or implicit, and either type can be used in a FOR loop. There are really two aspects to your question.
Why use an explicit cursor FOR loop over an implicit cursor FOR loop?
- Use an explicit cursor FOR loop when the query will be reused, otherwise an implicit cursor is preferred.
Why use a loop with a FETCH rather than a FOR loop that doesn’t have an explicit FETCH?
- Use a FETCH inside a loop when you need to bulk collect or when you need dynamic SQL.
Here is some useful information from the documentation.
Example of Implicit Cursor FOR LOOP
BEGIN
FOR vItems IN (
SELECT last_name
FROM employees
WHERE manager_id > 120
ORDER BY last_name
)
LOOP
DBMS_OUTPUT.PUT_LINE ('Name = ' || vItems.last_name);
END LOOP;
END;
/
Example of Explicit Cursor FOR LOOP
DECLARE
CURSOR c1 IS
SELECT last_name
FROM employees
WHERE manager_id > 120
ORDER BY last_name;
BEGIN
FOR vItems IN c1 LOOP
DBMS_OUTPUT.PUT_LINE ('Name = ' || vItems.last_name);
END LOOP;
END;
/
Implicit Cursor
An implicit cursor is a session cursor that is constructed and managed by PL/SQL. PL/SQL opens an implicit cursor every time you run a SELECT or DML statement. You cannot control an implicit cursor, but you can get information from its attributes.
An implicit cursor closes after its associated statement runs; however, its attribute values remain available until another SELECT or DML statement runs.
The implicit cursor attributes are: SQL%ISOPEN, SQL%FOUND, SQL%NOTFOUND, SQL%ROWCOUNT, SQL%BULK_ROWCOUNT, SQL%BULK_EXCEPTIONS
Explicit Cursor
An explicit cursor is a session cursor that you construct and manage. You must declare and define an explicit cursor, giving it a name and associating it with a query (typically, the query returns multiple rows). Then you can process the query result set in either of these ways:
Open the explicit cursor (with the OPEN statement), fetch rows from the result set (with the FETCH statement), and close the explicit cursor (with the CLOSE statement).
Use the explicit cursor in a cursor FOR LOOP statement (see "Query Result Set Processing With Cursor FOR LOOP Statements").
You cannot assign a value to an explicit cursor, use it in an expression, or use it as a formal subprogram parameter or host variable. You can do those things with a cursor variable (see "Cursor Variables").
Unlike an implicit cursor, you can reference an explicit cursor or cursor variable by its name. Therefore, an explicit cursor or cursor variable is called a named cursor.
Cursor FOR LOOP Statements
The cursor FOR LOOP statement lets you run a SELECT statement and then immediately loop through the rows of the result set. This statement can use either an implicit or explicit cursor.
I see that many developers are using explicit cursors instead of implicit cursors out of old habit. This because back in Oracle version 7 this was always the more efficient way to go. Nowadays there are generally the other way around. Specially with the optimizer that if needed may rewrite implicit cursor for loops to a bulk collect.