介绍:
DML / Select执行时的内存空间
游标(cursor)是指向这个内存空间的指针.
属性:
%ROWCOUNT %FOUND %NOTFOUNF %ISOPEN
隐式游标 / 显式游标(自定义游标名称)
SQL%ROWCOUNT / mycur%ROWCOUNT
显示游标的使用步骤:
1.定义游标
2.打开游标
3.操作游标
4.关闭游标
先建立一张表test_table:userid,user_name,password 3个字段
--显示打印,这里用的是显示游标
SET serveroutput ON;
--1.游标使用
DECLARE
v_name test_table.user_name%TYPE;
v_pwd test_table.PASSWORD%TYPE;
-- 1. 定义游标,把名字和SQL语句绑定
-- sql : select user_name,PASSWORD from test_table;
CURSOR test_cursor IS
SELECT user_name,PASSWORD FROM test_table;
BEGIN
-- 2. 打开游标:执行对应的SQL.取出记录
-- 等同于:rs = stmt.excuteQuery(sql);
OPEN test_cursor;
-- 3.循环取出游标中的结果,每次操作取一条,并使指针下移一条
loop
fetch test_cursor INTO v_name,v_pwd;
exit WHEN test_cursor%notfound;
dbms_output.put_line('name :'||v_name||' pwd :'||v_pwd);
END loop;
dbms_output.put_line('totle:'||test_cursor%rowcount);
-- 关闭游标,释放内存
CLOSE test_cursor;
end;
--2.定义有参数的游标
DECLARE
v_name test_table.user_name%TYPE;
v_pwd test_table.PASSWORD%TYPE;
-- 1. 定义游标
CURSOR test_cursor(v_id NUMBER) IS
SELECT user_name,PASSWORD FROM test_table where user_id = v_id;
BEGIN
-- 2. 打开游标,这里可以传入参数
OPEN test_cursor(2);
-- 3.操作游标,这里用主键查询,其实只有一条记录,你可以尝试其他的条件查询
loop
fetch test_cursor INTO v_name,v_pwd;
exit WHEN test_cursor%notfound;
dbms_output.put_line('name :'||v_name||' pwd :'||v_pwd);
END loop;
dbms_output.put_line('totle:'||test_cursor%rowcount);
-- 关闭游标,释放内存
CLOSE test_cursor;
end;
-- 3.游标类型和游标变量的使用
--类型语法:TYPE ref_type_name IS REF CURSOR [RETURN return_type];
--变量语法:cursor_name ref_type_name , 自定义变量名 + 游标类型
DECLARE
-- 定义一个动态(ref)游标类型,强类型(有RETURN)
TYPE test_cursor_type
IS REF CURSOR RETURN test_table%rowtype;
-- 定义一个游标变量,并使用刚才的类型
mycursor test_cursor_type;
-- 定义一个 行 类型,存放数据
myrecord test_table%rowtype;
BEGIN
-- 打开游标时,定义对应的SQL 语句。
OPEN mycursor FOR SELECT * FROM test_table;
loop
fetch mycursor INTO myrecord;
exit WHEN mycursor%notfound;
dbms_output.put_line(myrecord.user_id||' '||myrecord.user_name);
END loop;
CLOSE mycursor;
end;
--3.1 这是弱类型实例
DECLARE
input CHAR(1) := upper('&input');
--定义记录类型 和 记录变量
TYPE record_type IS record(
v_id NUMBER,
v_char varchar2(100)
);
myrecord record_type;
-- 定义游标类型和游标变量
TYPE cursor_type IS REF CURSOR; -- 无返回值,弱类型
mycursor cursor_type;
BEGIN
IF input = 'A' THEN
OPEN mycursor
FOR SELECT user_id,user_name FROM test_table;
elsif input = 'B' THEN
OPEN mycursor
FOR SELECT user_id,PASSWORD FROM test_table;
END IF;
fetch mycursor INTO myrecord;
while mycursor%found loop
dbms_output.put_line(myrecord.v_id||' '||myrecord.v_char);
fetch mycursor INTO myrecord;
--可以使用下面的循环方式
--loop
--fetch mycursor INTO myrecord;
--dbms_output.put_line(myrecord.v_id||' '||myrecord.v_char);
--exit WHEN mycursor%notfound;
END loop;
dbms_output.put_line('total :'||mycursor%rowcount);
CLOSE mycursor;
end;
--4 下面尝试一个简单的隐式游标:
BEGIN
INSERT INTO test_table VALUES(999,'123','123');
dbms_output.put_line(sql%rowcount);
end;
--或者
DECLARE
name varchar2(100);
BEGIN
SELECT user_name INTO NAME FROM test_table WHERE user_id = 999;
dbms_output.put_line(SQL%rowcount);
end;
--5 这个静态游标的使用,提前编译,速度更快。
DECLARE
CURSOR mycursor IS
SELECT * FROM test_table;
BEGIN
FOR myrecord IN mycursor loop
dbms_output.put_line(myrecord.user_id);
END loop;
end;
--5.1 和上面的一样,利用for 进行嵌套
BEGIN
FOR myrecord IN(SELECT * FROM test_table) loop
dbms_output.put_line(myrecord.user_id);
END loop;
end;
提示:
1. 游标相当于指针在数据空间排列的数据之间移动
2. 强弱类型只是看你是否要强制指定返回的类型,这里举例可能不好,可以根据参数查询不同的表
3. 显示游标都要用open cursor,而隐式不用,比如insert update DELETE select..into 等等
隐式游标不能用open fetch 等操作,属性%ISOPEN 会一直打开,语句结束就关闭
4.静态 动态游标的使用关键看你是否有不同的语句 或者参数。相当于更加灵活。 如果你仅仅一条SQL查询,就用静态游标就行了