Database Programming With PL/SQL 6-1: Practice Activities: Introduction To Explicit Cursors
Database Programming With PL/SQL 6-1: Practice Activities: Introduction To Explicit Cursors
Database Programming With PL/SQL 6-1: Practice Activities: Introduction To Explicit Cursors
Explicit cursors : Declared by the programmer for queries that return more than one row
Context area : An allocated memory area used to store the data processed by a SQL statement
Implicit cursor : Defined automatically by Oracle for all SQL DML statements, and for SELECT
statements that return only one row
OPEN statement : Statement that executes the query associated with the cursor, identifies the
active set, and positions the cursor pointer to the first row
FETCH statement : Statement that retrieves the current row and advances the cursor to the next
row either until there are no more rows or until a specified condition is met
Active set : The set of rows returned by a multiple row query in an explicit cursor operation
Try It / Solve It
1. In your own words, explain the difference between implicit and explicit cursors.
Implicit cursors are automatically created when select statements are executed, fetching one
row at a time. Where Explicit cursors needs to be defined explicitly by the user by providing a
name, explicit cursors can also fetch multiple rows.
2. Which SQL statement can use either an explicit or an implicit cursor, as needed?
when returning multiple rows, and when each row needs to be fetched manually.
4. Exercise using CURRENCIES tables:
A. Write a PL/SQL block to declare a cursor called currencies_cur. The cursor will
be used to read and display all rows from the CURRENCIES table. You will need to
retrieve currency_code and currency_name, ordered by ascending currency_name.
D. Add a statement to display the fetched row, and a statement to close the
currencies_cur cursor.
DECLARE
CURSOR currencies_cur IS
SELECT currency_code, currency_name FROM CURRENCIES
ORDER BY currency_name;
v_currency_code CURRENCIES.currency_code%TYPE;
v_currency_name CURRENCIES.currency_name%TYPE;
BEGIN
OPEN currencies_cur;
FETCH currencies_cur INTO v_currency_code, v_currency_name;
DBMS_OUTPUT.PUT_LINE(v_currency_code || ' ' || v_currency_name);
CLOSE currencies_cur;
END;
E. Your code so far displays only one row. Modify your code so that it fetches and
displays all the rows, using a LOOP and EXIT statement. Test your modified block. It
should fetch and display each row in the CURRENCIES table. If it doesn't, check that your
EXIT statement is in the correct place in t he code.
DECLARE
CURSOR currencies_cur IS
SELECT currency_code, currency_name FROM CURRENCIES
ORDER BY currency_name;
v_currency_code CURRENCIES.currency_code%TYPE;
v_currency_name CURRENCIES.currency_name%TYPE;
BEGIN
OPEN currencies_cur;
LOOP
FETCH currencies_cur INTO v_currency_code, v_currency_name;
EXIT WHEN currencies_cur%NOTFOUND;
DBMS_OUTPUT.PUT_LINE(v_currency_code || ' ' || v_currency_name);
END LOOP;
CLOSE currencies_cur;
END;
F. Write and test a PL/SQL block to read and display all the rows in the
COUNTRIES table for all countries in region 5 (South America region). For each
selected country, display the country_name, national_holiday_date, and
national_holiday_name. Display only those countries having a national holiday date that is
not null. Save your code (you will need it in the next practice).
DECLARE
CURSOR countries_cur IS
SELECT country_name, national_holiday_date, national_holiday_name
FROM COUNTRIES
WHERE region_id = 5 AND national_holiday_date IS NOT NULL;
v_country_name COUNTRIES.country_name%TYPE;
v_national_holiday_date COUNTRIES.national_holiday_date%TYPE;
v_national_holiday_name COUNTRIES.national_holiday_name%TYPE;
BEGIN
OPEN countries_cur;
LOOP
FETCH countries_cur INTO v_country_name, v_national_holiday_date, v_national_holiday_name;
EXIT WHEN countries_cur%NOTFOUND;
DBMS_OUTPUT.PUT_LINE(v_country_name || ' ' || v_national_holiday_date || ' ' ||
v_national_holiday_name);
END LOOP;
CLOSE countries_cur;
END;
1. Include the same number of variables in the INTO clause of the FETCH statement as columns in the
SELECT statement, and be sure that the data types are compatible.
2. Match each variable to correspond to the columns position in the cursor definition.
3. Use %TYPE to insure data types are compatible between variable and table.
6. Write a PL/SQL block to read and display the names of world regions, with a
count of the number of countries in each region. Include only those regions
having at least 10 countries. Order your output by ascending region name.
DECLARE
CURSOR region_cur IS
SELECT region_id, region_name FROM REGION
ORDER BY region_name;
CURSOR country_cur IS
SELECT region_id FROM COUNTRY;
v_region_id REGION.country_name%TYPE;
v_region_name REGION.region_name%TYPE;
v_country_region_id COUNTRY.region_id%TYPE;
v_counter NUMBER := 0;
BEGIN
OPEN region_cur;
LOOP
FETCH region_cur INTO v_region_id, v_region_name;
EXIT WHEN region_cur%NOTFOUND;
v_counter := 0;
OPEN country_cur;
LOOP
FETCH country_cur INTO v_country_region_id;
EXIT WHEN country_cur%NOTFOUND;
IF v_region_id = v_country_region_id THEN
v_counter := v_counter + 1;
END IF
END LOOP;
CLOSE country_cur;
IF v_counter >= 10 THEN
DBMS_OUTPUT.PUT_LINE('REGION NAME NUMBER OF COUNTRIES');
DBMS_OUTPUT.PUT_LINE('----------- -------------------');
DBMS_OUTPUT.PUT_LINE(v_region_name || ' ' || v_counter);
END IF;
END LOOP;
CLOSE region_cur;
END;