今年春节算是休了个长假,调整好心态,迎接新一年的挑战。
今天来说下Oracle中的循环迭代处理,因为从自己的博客统计中看到,不少网友都搜索了关键字"SQL FOR循环",所以打算在这里说下个人的理解。
PL/SQL也和我们常用的编程语言一样,提供了While、For等循环,我们建几个例子来说明演示下。
首先是While循环:
procedure loop_while
(
start_value in number,
end_value in number
)
is
current_value number := start_value;
begin while current_value <=end_value
loop
dbms_output.put_line('now number:' || current_value);
current_value:=current_value+1;
end loop;
end loop_while;
指定循环的初始值和结束值之后,就可以看到将这2个值之间的数字一行行打印出来了;当然只要while循环条件的求值结果是true,循环就会继续下去,
如果求值条件为false或者null,循环就会终止。这个循环条件每执行一次循环体之前都会先进行判断,因此while循环并不能保证循环体一定能被执行。
所以如果我们无法提前预知所需要巡检的次数的情况下,就可以使用While来进行循环处理。
For循环有2种,分别是数值型FOR循环和游标型FOR循环:
procedure loop_num_for
(
lowest in number,
highest in number
)
is
begin
FOR even_number in lowest .. highest --升序
loop
--处理非平滑增长的索引
if mod(even_number,2)=0
then
dbms_output.put_line('now number:' || even_number);
end if;
end loop;
end loop_num_for;
这种循环在开始的时候就已经知道循环的次数了,注意这里不需要声明循环索引,因为PL/SQL会自动隐式的用一个integer类型的局部变量作为它的循环索引;
如果要降序循环,必须加上reverse关键字,并且循环上边界和下边界的顺利无需改变:
loop
dbms_output.put_line('now number:' || even_number);
end loop;
另外需要说明的是,数值型FOR循环中,索引总是以1为单位递增或递减,所以如果我们的循环条件并非如此理想的平滑增长,我们就必须用一些逻辑代码或者技巧来
达到我们的目的。
如果我们需要对很多行记录做处理时,就可以使用游标型FOR循环:
procedure loop_cursor_for
is
begin
declare cursor userinfo_cur is select * from userinfo_table;
begin
FOR userinfo_rec in userinfo_cur
loop
dbms_output.put_line('username is:' || userinfo_rec.user_name);
end loop;
end;
end loop_cursor_for;
当游标中的所有记录都取出来后,FOR循环就会自动终止,这里不用显示OPEN、CLOSE游标,PL/SQL引擎会自动处理。
上面的循环语句都可以用EXIT 或者 EXIT WHEN来终止其循环,但最好不要这样做,因为这样可能会造成循环的逻辑出现问题,最终造成SQL代码难于跟踪和调试。
最后附上测试用的SQL:
--while循环
procedure loop_while(start_value in number, end_value in number) is
current_value number := start_value;
begin
while current_value <= end_value loop
dbms_output.put_line('now number:' || current_value);
current_value := current_value + 1;
end loop;
end loop_while;
--数值型For循环
procedure loop_num_for(lowest in number, highest in number) is
begin
FOR even_number in lowest .. highest
--升序
loop
首先一点,如果只是select 的话,Oracle是不会加任何锁的,也就是Oracle对 select 读到的数据不会有任何限制。
虽然这时候有可能另外一个进程正在修改表中的数据,并且修改的结果可能影响到你目前select语句的结果,但是因为没有锁,所以select结果为当前时刻表中记录的状态。
如果加入了for update, 则Oracle一旦发现(符合查询条件的)这批数据正在被修改,则不会发出该select语句查询,直到数据被修改结束(被commit),马上自动执行这个select语句。
同样,如果该查询语句发出后,有人需要修改这批数据(中的一条或几条),它也必须等到查询结束后(commit)后,才能修改。
for update nowait和 for update 都会对所查询到得结果集进行加锁,所不同的是,如果另外一个线程正在修改结果集中的数据,for update nowait 不会进行资源等待,只要发现结果集中有些数据被加锁,立刻返回 “ORA-00054错误,内容是资源正忙, 但指定以 NOWAIT 方式获取资源”。
PS:
for update 和 for update nowait 加上的是一个行级锁,也就是只有符合where条件的数据被加锁。如果仅仅用update语句来更改数据时,可能会因为加不上锁而没
有响应地、莫名其妙地等待,但如果在此之前,FOR UPDATE NOWAIT语句将要更改的数据试探性地加锁,就可以通过立即返回的错误提示而明白其中的道理,或许这就是FOR UPDATE和NOWAIT的意义之所在。
经过测试,以for update 或 for update nowait方式进行查询加锁,在select的结果集中,只要有任何一个记录在加锁,则整个结果集都在等待系统资源(如果是nowait,则抛出相应的异常)。
当有多个表关联查询时候,可以使用FOR UPDATE OF 子句来确定行锁定的特定表。
SELECT last_name, department_name FROM employees, departments
WHERE employees.department_id = departments.department_id
AND job_id = 'SA_MAN'
FOR UPDATE OF salary;
在sqldeveloper中一个session就是一个sql window
本文链接