Oracle数据库数据编程SQL<3.3 PL/SQL 游标>
游标(Cursor)是Oracle数据库中用于处理查询结果集的重要机制,它允许开发者逐行处理SQL语句返回的数据。

目录
一、游标基本概念
1. 游标定义
2. 游标分类
二、静态游标
(一)显式游标
【一】不带参数,普通的显示游标
1. 显式游标使用步骤
2. 语法
3. 显式游标的四个属性
4. 注意事项
5. %notfound 和普通循环一起用
6. %found 和while循环一起用
7. 基本示例
【二】带参数的显示游标
1. 语法结构
2. 示例代码
3. 练习
(二)隐式游标
1.隐式游标的四个属性
2. 示例代码
二、动态游标
【动态游标注意事项】
【强类型游标和弱类型游标区别】
【动态游标类型定义】
(一)强类型游标
(二)弱类型游标(SYS_REFCURSOR)
(三)动态游标
三、游标变量与批量处理
1. 游标变量
2. 批量提取(BULK COLLECT)
3. 批量处理与FORALL
四、游标最佳实践
五、高级游标技术
1. 可更新游标
2. 游标子查询
3. 游标表达式(12c+)
一、游标基本概念
1. 游标定义
游标是一个指向上下文区域的指针,用于处理SQL语句的执行结果。它提供了以下能力:
-
逐行访问结果集
-
跟踪当前处理的行
-
对结果集进行修改或删除操作
2. 游标分类
| 游标类型 | 描述 | 生命周期 | 控制方式 | |
|---|---|---|---|---|
| 静态 | 隐式游标 | Oracle自动为每条SQL语句创建 | 单条SQL执行期间 | Oracle自动管理 |
| 显式游标 | 开发者显式定义 | 从OPEN到CLOSE | 开发者手动控制 | |
| 动态 | REF游标 | 动态游标,运行时确定 | 灵活控制 | 开发者控制 |
二、静态游标

(一)显式游标
显示的游标:在declare的部分用is显示了的游标
【一】不带参数,普通的显示游标
1. 显式游标使用步骤
(1)声明游标:定义游标及其关联的SELECT语句
(2)打开游标:执行查询,填充结果集
(3)提取数据:从结果集中获取行数据
(4)关闭游标:释放资源
2. 语法
-- 1. 声明游标
CURSOR cursor_name [(parameters)][RETURN return_type]IS select_statement;-- 2. 打开游标
OPEN cursor_name [(parameters)];-- 3. 提取数据
FETCH cursor_name INTO variable_list;-- 4. 关闭游标
CLOSE cursor_name;-- 5. 举例
declare
cursor cur_name is select语句;---声明一个显示游标
begin open cur_name;--打开游标fetch cur_name into 变量;--赋值变量,提取记录dbms_output.put_line()---打印close cur_name;--关闭游标
end;
/
3. 显式游标的四个属性
| 属性 | 返回值 | 描述 | 说明 |
| cursor_name%FOUND | 布尔值 | 如果最近一次 FETCH返回行则为TRUE | 游标的指针是否有值(有)对 (没有) 错 |
| cursor_name%NOTFOUND | 布尔值 | 如果最近一次 FETCH未返回行则为TRUE | 游标的指针是否没值(有值)错, (没值)对,理论上可以返回空 在open之后fetch之前可以返回空 |
| cursor_name%ROWCOUNT | 数值 | 到目前为止已提取的行数 | 游标的指针已经指了几行,返回数值, 但是要赋给变量才能显示 返回最近一次从游标读取的数据 |
| cursor_name%ISOPEN | 布尔值 | 如果游标已打开则为TRUE | 判断是否打开游标(打开)对 (没有)错 |
4. 注意事项
首先声明一个游标,使用之前先打开游标,提取记录只能一行,可以多列
使用完游标要关闭游标,可以通过open打开游标继续使用
5. %notfound 和普通循环一起用
open→loop fetch→exit when %notfound→打印→end loop→close【举例1】declarecursor cur_a is select * from emp;---声明一个显示游标v_emp emp%rowtype;---声明变量v1 varchar2(20);---声明变量
beginopen cur_a;--打开游标loop/*普通循环*/ fetch cur_a into v_emp;/*赋值变量,抓取记录*/ exit when cur_a%notfound;/*和普通循环一起用*/ dbms_output.put_line(v_emp.ename); end loop;/*结束循环*/ close cur_a;/*关闭游标*/ end;
/-- LOOP循环语法
declare 部分;
beginloop 要执行的语句;exit when 退出的条件;end loop;
end;
注意事项:进入循环不需要条件
6. %found 和while循环一起用
open→fetch→while %found loop→打印→fetch→end loop→close【举例】
declarecursor cur_a is select * from emp;v_emp emp%rowtype;
beginopen cur_a;fetch cur_a into v_emp;while cur_a%found loopdbms_output.put_line(v_emp.ename);fetch cur_a into v_emp;end loop;close cur_a;
end;
7. 基本示例
DECLARE-- 1. 声明游标CURSOR emp_cursor ISSELECT employee_id, last_name, salaryFROM employeesWHERE department_id = 10;v_emp_id employees.employee_id%TYPE;v_name employees.last_name%TYPE;v_sal employees.salary%TYPE;
BEGIN-- 2. 打开游标OPEN emp_cursor;-- 3. 提取数据LOOPFETCH emp_cursor INTO v_emp_id, v_name, v_sal;EXIT WHEN emp_cursor%NOTFOUND;DBMS_OUTPUT.PUT_LINE(v_emp_id || ': ' || v_name || ', ' || v_sal);END LOOP;-- 4. 关闭游标CLOSE emp_cursor;
END;
/1、输出emp表工资前十名的员工姓名和薪资
declarecursor cur_1 isselect ename, salfrom (select ename, sal from emp order by sal desc)where rownum <= 10;v_n emp.ename%type;v_s emp.sal%type;
beginopen cur_1;loopfetch cur_1 into v_n, v_s;exit when cur_1%notfound;dbms_output.put_line('姓名:' || v_n || '薪资:' || v_s);end loop;close cur_1;
end;
/
-------------------
declarecursor cur_1 isselect ename, salfrom (select ename, sal from emp order by sal desc)where rownum <= 10;v_n emp.ename%type;v_s emp.sal%type;
beginopen cur_1;fetch cur_1 into v_n, v_s;while cur_1%found loopdbms_output.put_line('姓名:' || v_n || '薪资:' || v_s);fetch cur_1 into v_n, v_s;end loop;close cur_1;
end;
/
【二】带参数的显示游标
游标可以接受参数,使查询更加灵活:
1. 语法结构
CURSOR cursor_name (parameter1 datatype, parameter2 datatype, ...)IS select_statement;declarecursor cur_name(变量 类型) is select语句;---声明一个显示游标
begin open cur_name(变量);--打开游标fetch cur_name into 变量;--赋值变量,提取记录close cur_name;--关闭游标
end;
2. 示例代码
DECLARE-- 带参数的游标CURSOR emp_cursor (p_dept_id NUMBER, p_min_sal NUMBER) ISSELECT employee_id, last_name, salaryFROM employeesWHERE department_id = p_dept_idAND salary >= p_min_sal;-- 记录类型变量v_emp_record emp_cursor%ROWTYPE;
BEGIN-- 打开游标并传入参数OPEN emp_cursor(10, 5000);LOOPFETCH emp_cursor INTO v_emp_record;EXIT WHEN emp_cursor%NOTFOUND;DBMS_OUTPUT.PUT_LINE(v_emp_record.employee_id || ': ' || v_emp_record.last_name || ', ' || v_emp_record.salary);END LOOP;CLOSE emp_cursor;
END;
/declarecursor cur_1(v1 number) is select sal from emp where empno = v1; --声明一个带参数的显示游标v_s emp.sal%type; --负责接收的变量
beginopen cur_1(&a); ---(7788)即为(v1 number)fetch cur_1 into v_s;dbms_output.put_line(v_s);close cur_1;
end;
3. 练习
【1】输出工作是MANAGER的姓名、工资、;工作是SALESMAN的姓名、佣金;
工作是CLERK的姓名、入职日期。
declarecursor cur_2(v1 varchar2) isselect * from emp where job = v1;v_emp emp%rowtype;
beginopen cur_2('MANAGER');---loopfetch cur_2into v_emp;exit when cur_2%notfound;dbms_output.put_line(v_emp.ename ||' '|| v_emp.sal);end loop;close cur_2;open cur_2('SALESMAN');loopfetch cur_2into v_emp;exit when cur_2%notfound;dbms_output.put_line(v_emp.ename ||' '|| v_emp.comm);end loop;close cur_2;open cur_2('CLERK');loopfetch cur_2into v_emp;exit when cur_2%notfound;dbms_output.put_line(v_emp.ename ||' '|| to_char(v_emp.hiredate,'yyyy-mm-dd'));end loop;close cur_2;
end;
/【2】输出工作是MANAGER的姓名、工作、;工作是SALESMAN的姓名、佣金;
工作是CLERK的姓名、入职日期。
declarecursor cur_xx(v_1 emp.job%type) isselect * from emp where job = v_1;v_emp emp%rowtype;
beginopen cur_xx('MANAGER');loopfetch cur_xxinto v_emp;exit when cur_xx%notfound;dbms_output.put_line('员工姓名:'||v_emp.ename ||' '||'职位:'|| v_emp.job);end loop;close cur_xx;open cur_xx('SALESMAN');loopfetch cur_xxinto v_emp;exit when cur_xx%notfound;dbms_output.put_line('员工姓名:'||v_emp.ename ||' '||'工资:'|| v_emp.sal);end loop;close cur_xx;open cur_xx('CLERK');loopfetch cur_xxinto v_emp;exit when cur_xx%notfound;dbms_output.put_line('员工姓名:'||v_emp.ename ||' '||'入职日期:'||v_emp.hiredate);end loop;close cur_xx;
end;
/-----------------------------------------------------------------------------------------
输出名字中包含%的人
select * from emp where ename like '%'||v1||'%' or ename like'%'||v_2||'%'【3】打印名字中包含A的人数,包含E的平均工资,包含o的总工资
declarecursor cur_3(v1 varchar2) is---先让cur_3有了selecteselect count(sal), avg(sal), sum(sal)from empwhere ename like '%' || v1 || '%';----注意学习这种方法--声明一个带参数的显示游标cur_3v_2 number;v_3 number;v_4 number;
----------------添加负责接收的变量
beginopen cur_3('A');loopfetch cur_3---又从cur_3里提取值赋值给变量v_2,v_3,v_4into v_2, v_3, v_4;exit when cur_3%notfound;dbms_output.put_line(v_2);end loop;close cur_3;---------------open cur_3('E');loopfetch cur_3into v_2, v_3, v_4;exit when cur_3%notfound;dbms_output.put_line(v_3);end loop;close cur_3;----------------open cur_3('O');loopfetch cur_3into v_2, v_3, v_4;exit when cur_3%notfound;dbms_output.put_line(v_4);end loop;close cur_3;----------------
end;
(二)隐式游标
主要应用于增加删除更新数据,Oracle为每条DML语句自动创建隐式游标,当执行SQL语句的时候,这个游标是处理该语句的工作区域。在使用的时候要使用隐式游标的默认名称SQL。
1.隐式游标的四个属性
| 属性 | 返回 | 描述 | 说明 |
| cursor_name%FOUND | 布尔值 | 如果DML操作影响至少一行返回TRUE | 游标的游标中是否有值,返回最近一次的结果,成功(对)否则(错) |
| cursor_name%NOTFOUND | 布尔值 | 如果DML操作未影响任何行返回TRUE | 游标的游标中是否有值,返回最近一次的结果,成功(错) |
| cursor_name%ROWCOUNT | 数值 | 返回DML操作影响的行数 | 返回最近一次从游标中读取到的记录--数值类型 |
| cursor_name%ISOPEN | 布尔值 | 对隐式游标总是返回FALSE | 判断是否打开游标。永远返回错 |
补充:闪回不仅可以闪回删除前的数据,也可以返回之前某一时间点的数据
2. 示例代码
BEGIN-- 更新操作UPDATE employees SET salary = salary * 1.1 WHERE department_id = 10; -- 检查隐式游标属性IF SQL%FOUND THENDBMS_OUTPUT.PUT_LINE('更新了 ' || SQL%ROWCOUNT || ' 条记录');END IF;-- 删除操作DELETE FROM temp_employees WHERE employee_id = 9999;IF SQL%NOTFOUND THENDBMS_OUTPUT.PUT_LINE('未删除任何记录');END IF;
END;
----------------------------------------------------------------
----------------------------------------------------------------
/
begindelete from emp001 where deptno=10;--3--dbms_output.put_line('删除了'||sql%rowcount||'行');delete from emp001 where deptno=20;--5if sql%found thendbms_output.put_line('删除了'||sql%rowcount||'行') ;end if;end;
/
----------------------------------------------------------------
begin--dbms_output.put_line('删除了'||sql%rowcount||'行');delete from emp001 where deptno in (10,20);--5if sql%found thendbms_output.put_line('删除了'||sql%rowcount||'行') ;end if;
end;
/
----------------------------------------------------------------
----------------------------------------------------------------
declare
v_name csm_product.product_name%type;
begininsert into test_t values(1);if sql%found thendbms_output.put_line('收到影响的行数为:'||sql%rowcount);end if;rollback;
end;
----------------------------------------------------------------
----------------------------------------------------------------
declare
cursor v_cur is select * from test_t
beginif v_cur%isopen thendbms_output.put_line('游标已经打开');elsedbms_output.put_line('游标未打开');end if;open v_cur;if v_cur%isopen thendbms_output.put_line('游标已经打开');end if;close v_cur;insert into test_t valuse(1);if sql%found thendbms_output.put_line('执行成功,影响的行数:'||sql%rowcount)elsif sql%notfound thendbms_output.put_line('执行失败');end if;rollback;
end;
二、动态游标

【动态游标注意事项】
1、使用动态游标必须声明游标类型
2、只要列的格式相同,可以同时打开多表
【强类型游标和弱类型游标区别】
1、强类型游标有return,open时查询得到的结果要和return后的表的数据类型、结构、顺序一致。
2、没有return,open时查询的结果比较自由,fetch的时候into给的变量要和SQL查询的类型结构数量一致。
【动态游标类型定义】
TYPE cursor_type IS REF CURSOR [RETURN return_type];
(一)强类型游标
DECLARETYPE emp_cursor_type IS REF CURSOR RETURN employees%ROWTYPE;emp_cursor emp_cursor_type;v_emp employees%ROWTYPE;
BEGINOPEN emp_cursor FOR SELECT * FROM employees WHERE department_id = 20;LOOPFETCH emp_cursor INTO v_emp;EXIT WHEN emp_cursor%NOTFOUND;DBMS_OUTPUT.PUT_LINE(v_emp.employee_id || ': ' || v_emp.last_name);END LOOP;CLOSE emp_cursor;
END;
/declaretype cur_name_ref is ref cursor return emp%rowtype; --emp可以自定义,只是声明了一个类型cur_1 cur_name_ref;v_emp emp%rowtype;
begin
--------------------------------------------------------------------open cur_1 for select * from emp;-------------同时打开多表---------fetch cur_1into v_emp;dbms_output.put_line(v_emp.deptno || v.emp.ename || v_emp.job);close cur_1;
--------------------------------------------------------------------open cur_1 for select * from emp001;---------同时打开多表----------fetch cur_1into v_emp;dbms_output.put_line(v_emp.deptno || v.emp.ename || v_emp.job);close cur_1;
end;
(二)弱类型游标(SYS_REFCURSOR)
DECLAREemp_cursor SYS_REFCURSOR;v_emp_id employees.employee_id%TYPE;v_emp_name employees.last_name%TYPE;
BEGIN-- 打开第一个查询OPEN emp_cursor FOR SELECT employee_id, last_name FROM employees WHERE department_id = 10;DBMS_OUTPUT.PUT_LINE('部门10员工:');LOOPFETCH emp_cursor INTO v_emp_id, v_emp_name;EXIT WHEN emp_cursor%NOTFOUND;DBMS_OUTPUT.PUT_LINE(v_emp_id || ': ' || v_emp_name);END LOOP;CLOSE emp_cursor;-- 重用游标执行不同查询OPEN emp_cursor FOR SELECT department_id, department_name FROM departments;DBMS_OUTPUT.PUT_LINE('所有部门:');LOOPFETCH emp_cursor INTO v_emp_id, v_emp_name; -- 重用变量EXIT WHEN emp_cursor%NOTFOUND;DBMS_OUTPUT.PUT_LINE(v_emp_id || ': ' || v_emp_name);END LOOP;CLOSE emp_cursor;
END;
/declaretype cur_name_ref is ref cursor;---没有return,和强类型的区别就在于此cur_name cur_name_ref;v_emp emp%rowtype;
beginopen cur_name for select * from emp;-------------同时打开多表---------------fetch cur_nameinto v_emp;dbms_output.put_line(v_emp.deptno || v.emp.ename || v_emp.job);close cur_1;---注意游标开一次,关一次,不然一直开着耗内存。open cur_name for select * from emp001;--------同时打开多表---------------fetch cur_nameinto v_emp;dbms_output.put_line(v_emp.deptno || v.emp.ename || v_emp.job);close cur_1;
end;【练习题】
1、输出工作是MANAGER的姓名、工资;工作是SALESMAN的姓名、佣金;
工作是CLERK的姓名、入职日期。
declaretype cur_1_ref is ref cursor; ---没有returncur_1 cur_1_ref;v1 varchar2(20);v2 number;v3 date;
beginopen cur_1 forselect ename, sal from emp where job = 'MANAGER';loopfetch cur_1into v1, v2;exit when cur_1%notfound;dbms_output.put_line(v1 || v2);end loop;open cur_1 forselect ename, sal from emp where job = 'SALESMAN';loopfetch cur_1into v1, v2;exit when cur_1%notfound;dbms_output.put_line(v1 || v2);end loop;open cur_1 forselect ename, hiredate from emp where job = 'CLERK';loopfetch cur_1into v1, v3;exit when cur_1%notfound;dbms_output.put_line(v1 || v3);end loop;
end;
/--用弱类型游标,打印emp名字中包含A的人数,dept部门编号的平均数
--salgrade 第三等级的hisal
declaretype cur_name_ref is ref cursor;cur_gam cur_name_ref;v_1 number;
beginopen cur_gam forselect count(ename) from emp where ename like '%A%';----单列单行不用循环fetch cur_gaminto v_1;dbms_output.put_line('名字包含A的人数:' || v_1);close cur_gam;open cur_gam forselect avg(deptno) from dept;----单列单行不用循环fetch cur_gaminto v_1;dbms_output.put_line('部门编号的平均数:' || v_1);close cur_gam;open cur_gam forselect hisal from salgrade where grade = 3;----单列单行不用循环fetch cur_gaminto v_1;dbms_output.put_line('第三等级的hisal:' || v_1);close cur_gam;
end;
--------如果多行多列,加入循环的写法
declaretype cur_name_ref is ref cursor;cur_gam cur_name_ref;v_1 number;
beginopen cur_gam forselect count(ename) from emp where ename like '%A%';loopfetch cur_gaminto v_1;exit when cur_gam%notfound;dbms_output.put_line('名字包含A的人数:'||v_1);end loop;close cur_gam;
-----------------------open cur_gam forselect avg(deptno) from dept;loopfetch cur_gaminto v_1;exit when cur_gam%notfound;dbms_output.put_line('部门编号的平均数:'||v_1);end loop;close cur_gam;
-----------------------------open cur_gam forselect hisal from salgrade where grade=3;loopfetch cur_gaminto v_1;exit when cur_gam%notfound;dbms_output.put_line('第三等级的hisal:'||v_1);end loop;close cur_gam;
end;
(三)动态游标
【语法】
declarecur_name sys_refcursor;v1 number;v2 varchar2(20);
beginopen cur_name forselect empno, ename from emp where empno = 7788;fetch cur_nameinto v1, v2;dbms_output.put_line(v1 || v2);close cur_name;
--------------open cur_name forselect empno, ename from emp where deptno = 20;fetch cur_nameinto v1, v2;dbms_output.put_line(v1 || v2);close cur_name;
end;【练习题】
1、打印10部门的人员姓名和部门地址,打印20部门的工资和工资等级
declarecur_1 sys_refcursor;ve varchar2(20);vc varchar2(20);vs number;vg number;
beginopen cur_1 forselect e.ename,d.loc from emp e,dept d where e.deptno=d.deptno and d.deptno=10;loopfetch cur_1into ve, vc;exit when cur_1%notfound;dbms_output.put_line(ve ||' '|| vc);end loop;open cur_1 forselect e.sal, s.grade from emp e,salgrade s where e.sal between s.losal and s.hisal and e.deptno=20;loopfetch cur_1into vs, vg;exit when cur_1%notfound;dbms_output.put_line(vs ||' '|| vg);end loop;close cur_1;
end;
/2、用动态游标里面的动态游标,打印emp名字中包含A的人数,dept部门编号的平均数,salgrade 第三等级的hisal
declarecur_2 sys_refcursor;v1 number;
beginopen cur_2 forselect count(ename) from emp where ename like '%A%';fetch cur_2into v1;dbms_output.put_line('名字包含A的人数:' || v1);close cur_2;
----------------------open cur_2 forselect avg(deptno) from dept;fetch cur_2into v1;dbms_output.put_line('部门编号的平均数:' || v1);close cur_2;
-----------------------open cur_2 forselect hisal from salgrade where grade = 3;fetch cur_2into v1;dbms_output.put_line('第三等级的hisal:' || v1);close cur_2;
end;
/

三、游标变量与批量处理
1. 游标变量
DECLARETYPE emp_cursor_type IS REF CURSOR;emp_cursor emp_cursor_type;PROCEDURE process_employees (p_cursor IN emp_cursor_type) ISv_emp employees%ROWTYPE;BEGINLOOPFETCH p_cursor INTO v_emp;EXIT WHEN p_cursor%NOTFOUND;DBMS_OUTPUT.PUT_LINE(v_emp.employee_id || ': ' || v_emp.last_name);END LOOP;END;
BEGINOPEN emp_cursor FOR SELECT * FROM employees WHERE department_id = 10;process_employees(emp_cursor);CLOSE emp_cursor;
END;
/
2. 批量提取(BULK COLLECT)
DECLARECURSOR emp_cursor ISSELECT employee_id, last_name, salaryFROM employeesWHERE department_id = 10;-- 定义集合类型TYPE emp_id_array IS TABLE OF employees.employee_id%TYPE;TYPE name_array IS TABLE OF employees.last_name%TYPE;TYPE sal_array IS TABLE OF employees.salary%TYPE;v_ids emp_id_array;v_names name_array;v_sals sal_array;
BEGINOPEN emp_cursor;-- 批量提取数据FETCH emp_cursor BULK COLLECT INTO v_ids, v_names, v_sals;CLOSE emp_cursor;-- 处理批量数据FOR i IN 1..v_ids.COUNT LOOPDBMS_OUTPUT.PUT_LINE(v_ids(i) || ': ' || v_names(i) || ', ' || v_sals(i));END LOOP;
END;
/
3. 批量处理与FORALL
DECLARETYPE id_array IS TABLE OF employees.employee_id%TYPE;TYPE sal_array IS TABLE OF employees.salary%TYPE;v_ids id_array := id_array(101, 102, 103, 104, 105);v_new_sals sal_array;
BEGIN-- 批量查询SELECT salary BULK COLLECT INTO v_new_salsFROM employeesWHERE employee_id IN (SELECT COLUMN_VALUE FROM TABLE(v_ids));-- 批量更新FORALL i IN 1..v_ids.COUNTUPDATE employeesSET salary = v_new_sals(i) * 1.1WHERE employee_id = v_ids(i);COMMIT;DBMS_OUTPUT.PUT_LINE('成功更新 ' || SQL%ROWCOUNT || ' 条记录');
END;
/
四、游标最佳实践
1. 及时关闭游标:避免资源泄漏
BEGINOPEN emp_cursor;-- 处理数据
EXCEPTIONWHEN OTHERS THENIF emp_cursor%ISOPEN THENCLOSE emp_cursor;END IF;RAISE;
END;
2. 使用游标FOR循环:简化代码,自动处理打开/关闭
3. 批量操作:对大结果集使用BULK COLLECT和FORALL
4. 参数化游标:提高代码重用性
5. 限制返回行数:避免内存问题
FETCH emp_cursor BULK COLLECT INTO v_emps LIMIT 1000;
6. 使用合适的作用域:在包中定义常用游标
7. 性能监控:检查游标SQL的执行计划
五、高级游标技术
1. 可更新游标
DECLARECURSOR emp_cursor ISSELECT employee_id, last_name, salaryFROM employeesWHERE department_id = 10FOR UPDATE OF salary NOWAIT;v_raise_percent NUMBER := 0.1;
BEGINFOR emp_rec IN emp_cursor LOOPUPDATE employeesSET salary = salary * (1 + v_raise_percent)WHERE CURRENT OF emp_cursor;END LOOP;COMMIT;
END;
/
2. 游标子查询
DECLARECURSOR dept_cursor ISSELECT d.department_id, d.department_name,CURSOR(SELECT employee_id, last_nameFROM employeesWHERE department_id = d.department_id) AS emp_cursorFROM departments dWHERE d.location_id = 1700;v_emp_cursor SYS_REFCURSOR;v_emp_id employees.employee_id%TYPE;v_emp_name employees.last_name%TYPE;
BEGINFOR dept_rec IN dept_cursor LOOPDBMS_OUTPUT.PUT_LINE('部门: ' || dept_rec.department_name);v_emp_cursor := dept_rec.emp_cursor;LOOPFETCH v_emp_cursor INTO v_emp_id, v_emp_name;EXIT WHEN v_emp_cursor%NOTFOUND;DBMS_OUTPUT.PUT_LINE(' ' || v_emp_id || ': ' || v_emp_name);END LOOP;CLOSE v_emp_cursor;END LOOP;
END;
/
3. 游标表达式(12c+)
DECLARECURSOR dept_cursor ISSELECT d.department_id, d.department_name,CURSOR(SELECT e.employee_id, e.last_nameFROM employees eWHERE e.department_id = d.department_id) AS emp_curFROM departments d;
BEGINFOR dept_rec IN dept_cursor LOOPDBMS_OUTPUT.PUT_LINE('部门: ' || dept_rec.department_name);FOR emp_rec IN dept_rec.emp_cur LOOPDBMS_OUTPUT.PUT_LINE(' 员工: ' || emp_rec.last_name);END LOOP;END LOOP;
END;
/
游标是Oracle PL/SQL中处理结果集的核心机制,掌握各种游标技术可以显著提高数据库应用程序的效率和灵活性。
相关文章:
Oracle数据库数据编程SQL<3.3 PL/SQL 游标>
游标(Cursor)是Oracle数据库中用于处理查询结果集的重要机制,它允许开发者逐行处理SQL语句返回的数据。 目录 一、游标基本概念 1. 游标定义 2. 游标分类 二、静态游标 (一)显式游标 【一】不带参数,普通的显示游标 1. 显式…...
畅享电脑流畅运行:深度卸载、智能监视与空间释放
软件介绍 在数字化办公与娱乐高度融合的当下,电脑承载着我们诸多重要任务,然而,随着软件的频繁安装与卸载,系统逐渐被各种顽固软件及其残留 “拖垮”,运行速度变慢、磁盘空间告急等问题接踵而至。别愁,今天…...
R --- Error in library(***) : there is no package called ‘***’ (服务器非root用户)
步骤 步骤一:在自己目录下创建R包安装路径步骤二:配置用户本地的R库路径步骤三:安装缺失的包(在终端)步骤四:验证安装 步骤一:在自己目录下创建R包安装路径 mkdir -p ~/R_libs步骤二࿱…...
Visual Studio Code 无法打开源文件解决方法
🌈 个人主页:Zfox_ 🔥 系列专栏:Linux 🔥 系列专栏:C从入门到精通 目录 一:🔥 突发状况 二:🔥 共勉 一:🔥 突发状况 🐬…...
核函数(机器学习深度学习)
一、核函数的基本概念 核函数(Kernel Function) 是机器学习中处理非线性问题的核心工具,通过隐式映射将数据从原始空间转换到高维特征空间,从而在高维空间中实现线性可分或线性建模。其数学本质是计算两个样本在高维空间中的内积…...
【工具】BioPred一个用于精准医疗中生物标志物分析的 R 软件包
介绍 R 语言包 BioPred 提供了一系列用于精准医疗中的亚组分析和生物标志物分析的工具。它借助极端梯度提升(XGBoost)算法,并结合倾向得分加权和 A 学习方法,帮助优化个体化治疗规则,从而简化亚组识别过程。BioPred 还…...
【银河麒麟系统常识】命令:dotnet --list-sdks(列出已安装的 .NET SDK 版本)
命令: dotnet --list-sdks 功能 列出当前系统中所有已安装的 .NET SDK 版本; 返回值规则 # 1. 格式:<版本号>[<安装路径>]; # 2. 排序:按版本号从低到高排序;示例...
【深度学习】不管理论,入门从手写数字识别开始
1. 环境安装 学习深度学习,开发语言是Python。Python开发工具有很多。其中 anaconda vscode的Python开发环境很好用,建议使用这个组合。 编写手写数字识别测试代码,需要在使用Anaconda安装以下4个库: NumpyScipymatplotlibsci…...
3.使用epoll实现单线程并发服务器
目录 1. epoll的概述 2. 多线程与epoll的处理流程 2.1 多线程处理流程 2.2 epoll处理流程 3. epoll与多线程的比较 4. epoll的操作函数 4.1 epoll_create() 4.2 epoll_ctl() 4.3 epoll_wait() 5. 示例代码 6. epoll的工作模式 7. 使用O_NONBLOCK防止阻塞 8.运行代…...
关于JVM和OS中的栈帧的区别和内存浅析
关于JVM和OS中的栈帧的区别和内存浅析 刚看了黑马JVM中的栈帧的讲解,感觉和自己理解的栈帧有一定出入,查询资料研究了一下发现的确有天壤之别,可惜黑马并没有讲。 故写下这篇文章巩固一下, OS的栈帧: OS的栈帧会在调用一个函…...
拥抱健康生活,开启养生之旅
在快节奏的现代生活中,健康养生愈发重要。它不仅能让我们拥有强健体魄,还能提升生活质量。 均衡饮食是养生的基石。多吃蔬菜和水果,它们富含维生素与膳食纤维。比如西兰花,堪称 “蔬菜皇冠”,不仅含有丰富的维生素 …...
测试用例管理工具
一、免费/开源工具 TestLink 适用场景:传统手工测试团队,需基础用例管理与测试计划跟踪。 关键功能:用例分层管理、执行结果记录、基础报告生成。 局限:界面陈旧,自动化集成需插件支持。 Kiwi TCMS 适用场景࿱…...
visual studio 2017配置QT5.9.4环境
前提是已经安装完毕vs 2017以及QT5.9.4,然后再进行下列的操作 一 环境配置 修改成如下所示,然后关闭vs 打开浏览器,搜索网站download.qt.io 如果2.4.1版本出现问题,可以换版本,如2.3.1,2.7.1都比较稳定 …...
基于EFISH-SBC-RK3576的无人机智能飞控与数据存储方案
一、方案背景 民用无人机在电力巡检、农业植保、应急救援等领域快速普及,但传统方案面临多协议设备兼容性差、野外环境数据易丢失、复杂电磁干扰三大痛点。 电鱼智能推出EFISH-SBC-RK3576,可集成双冗余总线接口与工业级加固存储&#x…...
c++的特性——多态
目录 概念 多态实现条件 虚函数 虚函数的重写/覆盖 练习题 析构函数的重写 override和final关键字 重载/隐藏/重载的区别 纯虚函数和抽象类 多态 虚函数表指针 多态的原理 动态绑定与静态绑定 虚函数表总结 前面学习了C的三个特性中的两个特性,今天我们…...
MySQL基础语法DDLDML
目录 #1.创建和删除数据库 #2.如果有lyt就删除,没有则创建一个新的lyt #3.切换到lyt数据库下 #4.创建数据表并设置列及其属性,name是关键词要用name包围 编辑 #5.删除数据表 #5.查看创建的student表 #6.向student表中添加数据,数据要与列名一一对应 #7.查询studen…...
性能测试理论基础-性能指标及jmeter中的指标
1、什么是性能测试 通过一定的手段,在多并发下情况下,获取被测系统的各项性能指标,验证被测系统在高并发下的处理能力、响应能力,稳定性等,能否满足预期。定位性能瓶颈,排查性能隐患,保障系统的质量,提升用户体验。 2、什么样的系统需要做性能测试 用户量大,页面访问…...
Postman CORS 测试完全指南:轻松模拟跨域请求,排查 CORS 相关问题
在使用 Postman 进行 API 测试时,通常不会遇到跨域问题,因为 Postman 是一个独立的客户端应用程序,不同于在浏览器中运行的 JavaScript 代码,它没有同源策略(SOP)的限制。跨域资源共享(CORS&…...
iOS抓包-charles和Stream
简单介绍几种抓包工具 1、Charles Charles是一款流行的跨平台HTTP代理软件,常用于Web调试,它可以帮助你在开发过程中检查、修改或模拟HTTP/HTTPS请求和响应。以下是如何在iOS设备上使用Charles进行抓包的基本步骤: 第一步:安装…...
三个核心文件:src\App.vue文件,index.html文件,src\main.js文件 的关系与运行流程解析(通俗形象)
一、三个文件的角色定位 用生活比喻理解它们的关系: index.html → “空房子” 像一栋毛坯房,只有基本的墙面和预留的插座(空白的HTML结构)。它的作用是提供一个容器,告诉Vue:“请把装修好的房间…...
云原生系列-K8S实战
K8S实战 1. K8S 资源创建方式2. NameSpace 资源创建3. Pod4. Deployment5. Service6. Ingress7. 存储抽象1. 环境准备2. PV&PVC1) 创建PV池2) PVC创建与绑定 3. ConfigMap 抽取应用配置,并且可以自动更新1) redis 示例2) 创建…...
从责任链模式聊到aware接口
从责任链模式聊到aware接口 责任链是什么? 责任链模式是一种行为型设计模式,将多个对象连接成一条链,并且沿着这条链传递请求,让多个对象都有机会处理这个请求,请求会顺着链传递,直到某个对象处理它为止。…...
ArcGIS地理信息系统空间分析实验教程学习
ArcGIS 作为地理信息系统领域的经典软件,以其强大的功能和广泛的应用场景,成为了众多学者、研究人员和专业人士的首选工具。它不仅可以高效地处理和可视化地理空间数据,还能通过复杂的空间分析模型,揭示地理现象背后的规律和趋势。…...
【3天!!!从0-1完成自动化集成平台开发--Cursor AI赋能0代码基础测试工程师开发平台-亲测有效-保姆级】
利用Cursor AI 赋能测试工程师从0-1开发自动化集成平台 ——含框架设计、实例代码与CI/CD集成 一、技术选型与框架设计1.1 核心框架选择1.2 整体架构图二、从0到1开发步骤2.1 初始化项目2.2 核心模块开发模块1:测试用例管理(Python)模块2:测试执行引擎(pytest)三、实战案…...
Compose 实践与探索十七 —— 多指手势与自定义触摸反馈
上一节我们讲了滑动的手势识别以及嵌套滑动,二者都属于触摸反馈这个大的范畴内的知识。本节我们将深入触摸反馈这个话题,讲一讲多指手势的识别与完全自定义的触摸反馈的实现。 1、多指手势 多指手势可以分为两类: 利用 API 处理预设好的手…...
Spring Boot 整合 ElasticJob 分布式任务调度教程
精心整理了最新的面试资料和简历模板,有需要的可以自行获取 点击前往百度网盘获取 点击前往夸克网盘获取 Spring Boot 整合 ElasticJob 分布式任务调度教程 一、ElasticJob 简介 ElasticJob 是当当网开源的分布式任务调度解决方案,支持: …...
Go 语言规范学习(6)
文章目录 StatementsTerminating statementsEmpty statementsLabeled statementsExpression statementsSend statementsIncDec statementsAssignment statementsIf statementsSwitch statementsExpression switchesType switches For statementsFor statements with single con…...
centos8上实现lvs集群负载均衡nat模式
1.背景: 个人(菜鸟)学习笔记,学点记下来,给未来的自己看。高手看了也请多指点。 按照课程讲,lvs是我国大神开发的负载均衡程序,被收录进内核,只要安装时内核里有它,它就…...
深度学习篇---模型参数调优
文章目录 前言一、Adam学习(lr)1. 默认学习率2. 较小的学习率模型复杂数据集规模小 3. 较大的学习率模型简单训练初期 4. 学习率衰减策略固定步长衰减指数衰减 二、训练轮数(epoch)1. 经验值设定小数据集与简单模型大数据集和复杂…...
影响HTTP网络请求的因素
影响 HTTP 网络请求的因素 1. 带宽 2. 延迟 浏览器阻塞:浏览器会因为一些原因阻塞请求,浏览器对于同一个域名,同时只能有4个连接(这个根据浏览器内核不同可能会有所差异),超过浏览器最大连接数限制&…...
