这篇文章主要介绍了Oracle中游标Cursor基本用法详解,还是比较全面的,具有一定参考价值,需要的朋友可以了解下。

查询

  SELECT语句用于从数据库中查询数据,当在PL/SQL中使用SELECT语句时,要与INTO子句一起使用,查询的
返回值被赋予INTO子句中的变量,变量的声明是在DELCARE中。SELECT INTO语法如下:

?

1

2

3

4

SELECT [DISTICT|ALL]{*|column[,column,...]}

INTO (variable[,variable,...] |record)

FROM {table|(sub-query)}[alias]

WHERE............

  PL/SQL中SELECT语句只返回一行数据。如果超过一行数据,那么就要使用显式游标(对游标的讨论我们将
在后面进行),INTO子句中要有与SELECT子句中相同列数量的变量。INTO子句中也可以是记录变量。

%TYPE属性

  在PL/SQL中可以将变量和常量声明为内建或用户定义的数据类型,以引用一个列名,同时继承他的数据类
型和大小。这种动态赋值方法是非常有用的,比如变量引用的列的数据类型和大小改变了,如果使用了%TYPE,
那么用户就不必修改代码,否则就必须修改代码。

例:

?

1

2

v_empno SCOTT.EMP.EMPNO%TYPE;

v_salary EMP.SALARY%TYPE;

  不但列名可以使用%TYPE,而且变量、游标、记录,或声明的常量都可以使用%TYPE。这对于定义相同数据类
型的变量非常有用。

?

1

2

3

4

5

6

7

8

9

10

11

12

DELCARE

 V_A NUMBER(5):=10;

V_B V_A%TYPE:=15;

V_C V_A%TYPE;

BEGIN

DBMS_OUTPUT.PUT_LINE

('V_A='||V_A||'V_B='||V_B||'V_C='||V_C);

END

SQL>/

V_A=10 V_B=15 V_C=

PL/SQL procedure successfully completed.

SQL>

其他DML语句

其它操作数据的DML语句是:INSERT、UPDATE、DELETE和LOCK TABLE,这些语句在PL/SQL中的语法与在SQL中
的语法相同。我们在前面已经讨

论过DML语句的使用这里就不再重复了。在DML语句中可以使用任何在DECLARE部分声明的变量,如果是嵌套
块,那么要注意变量的作用范围。

例:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

CREATE OR REPLACE PROCEDURE FIRE_EMPLOYEE (pempno in number)

AS

v_ename EMP.ENAME%TYPE;

BEGIN

SELECT ename INTO v_ename

FROM emp

WHERE empno=p_empno;

INSERT INTO FORMER_EMP(EMPNO,ENAME)

VALUES (p_empno,v_ename);

DELETE FROM emp

WHERE empno=p_empno;

UPDATE former_emp

SET date_deleted=SYSDATE

WHERE empno=p_empno;

EXCEPTION

WHEN NO_DATA_FOUND THEN

DBMS_OUTPUT.PUT_LINE('Employee Number Not Found!');

END

  DML语句的结果

当执行一条DML语句后,DML语句的结果保存在四个游标属性中,这些属性用于控制程序流程或者了解程序
的状态。当运行DML语句时,PL/SQL打开一个内建游标并处理结果,游标是维护查询结果的内存中的一个区域,
游标在运行DML语句时打开,完成后关闭。隐式游标只使用SQL%FOUND,

SQL%NOTFOUND,SQL%ROWCOUNT三个属性.SQL%FOUND,SQL%NOTFOUND是布尔值,SQL%ROWCOUNT是整数值。

SQL%FOUND和SQL%NOTFOUND

在执行任何DML语句前SQL%FOUND和SQL%NOTFOUND的值都是NULL,在执行DML语句后,SQL%FOUND的属性值将是:

  . TRUE :INSERT
  . TRUE :DELETE和UPDATE,至少有一行被DELETE或UPDATE.
  . TRUE :SELECT INTO至少返回一行

当SQL%FOUND为TRUE时,SQL%NOTFOUND为FALSE。

SQL%ROWCOUNT

在执行任何DML语句之前,SQL%ROWCOUNT的值都是NULL,对于SELECT INTO语句,如果执行成功
,SQL%ROWCOUNT的值为1,如果没有成功,SQL%ROWCOUNT的值为0,同时产生一个异常NO_DATA_FOUND.

SQL%ISOPEN

SQL%ISOPEN是一个布尔值,如果游标打开,则为TRUE, 如果游标关闭,则为FALSE.对于隐式游标而言SQL%
ISOPEN总是FALSE,这是因为隐式游

标在DML语句执行时打开,结束时就立即关闭。

事务控制语句

事务是一个工作的逻辑单元可以包括一个或多个DML语句,事物控制帮助用户保证数据的一致性。如果事务控制逻辑单元中的任何一个DML
语句失败,那么整个事务都将回滚,在PL/SQL中用户可以明确地使用COMMIT、ROLLBACK、SAVEPOINT以及
SET TRANSACTION语句。

COMMIT语句终止事务,永久保存数据库的变化,同时释放所有LOCK,ROLLBACK终止现行事务释放所有LOCK,
但不保存数据库的任何变化,SAVEPOI

NT用于设置中间点,当事务调用过多的数据库操作时,中间点是非常有用的,SET TRANSACTION用于设置事
务属性,比如read-write和隔离级等。

显式游标

查询返回结果超过一行时,就需要一个显式游标,此时用户不能使用select into语句。PL/SQL管理隐式
游标,当查询开始时隐式游标打开,查询结束时隐式游标自动关闭。显式游标在PL/SQL块的声明部分声明,在执行部分或异常处理部分打开,取数据,关闭。

使用游标

这里要做一个声明,我们所说的游标通常是指显式游标,因此从现在起没有特别指明的情况,我们所说的
游标都是指显式游标。要在程序中使用游标,必须首先声明游标。

声明游标

语法:

CURSOR cursor_name IS select_statement;

在PL/SQL中游标名是一个未声明变量,不能给游标名赋值或用于表达式中。

例:

?

1

2

3

4

5

6

7

DELCARE

CURSOR C_EMP IS SELECT empno,ename,salary

FROM emp

WHERE salary>2000

ORDER BY ename;

........

BEGIN

  在游标定义中SELECT语句中不一定非要表可以是视图,也可以从多个表或视图中选择的列,甚至可以使用*
来选择所有的列 。

打开游标

使用游标中的值之前应该首先打开游标,打开游标初始化查询处理。打开游标的语法是:

?

1

OPEN cursor_name

 cursor_name是在声明部分定义的游标名。

例:

?

1

OPEN C_EMP;

  关闭游标

语法:

?

1

CLOSE cursor_name

例:

?

1

CLOSE C_EMP;

从游标提取数据

从游标得到一行数据使用FETCH命令。每一次提取数据后,游标都指向结果集的下一行。语法如下:

FETCH cursor_name INTO variable[,variable,...]

对于SELECT定义的游标的每一列,FETCH变量列表都应该有一个变量与之相对应,变量的类型也要相同。

例:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

  SET SERVERIUTPUT ON

  DECLARE

  v_ename EMP.ENAME%TYPE;

  v_salary EMP.SALARY%TYPE;

  CURSOR c_emp IS SELECT ename,salary FROM emp;

  BEGIN

  OPEN c_emp;

  FETCH c_emp INTO v_ename,v_salary;

DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename ||'is'|| v_salary);

  FETCH c_emp INTO v_ename,v_salary;

  DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename ||'is'|| v_salary);

  FETCH c_emp INTO v_ename,v_salary;

  DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename ||'is'|| v_salary);

  CLOSE c_emp;

  END

这段代码无疑是非常麻烦的,如果有多行返回结果,可以使用循环并用游标属性为结束循环的条件,以这
种方式提取数据,程序的可读性和简洁性都大为提高,下面我们使用循环重新写上面的程序:

?

1

2

3

4

5

6

7

8

9

10

11

12

SET SERVERIUTPUT ON

DECLARE

v_ename EMP.ENAME%TYPE;

v_salary EMP.SALARY%TYPE;

CURSOR c_emp IS SELECT ename,salary FROM emp;

BEGIN

OPEN c_emp;

LOOP

FETCH c_emp INTO v_ename,v_salary;

EXIT WHEN c_emp%NOTFOUND;

DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename ||'is'|| v_salary);

END

  记录变量

定义一个记录变量使用TYPE命令和%ROWTYPE,关于%ROWsTYPE的更多信息请参阅相关资料。

记录变量用于从游标中提取数据行,当游标选择很多列的时候,那么使用记录比为每列声明一个变量要方
便得多。

当在表上使用%ROWTYPE并将从游标中取出的值放入记录中时,如果要选择表中所有列,那么在SELECT子句
中使用*比将所有列名列出来要安全得多。

例:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

SET SERVERIUTPUT ON

DECLARE

R_emp EMP%ROWTYPE;

CURSOR c_emp IS SELECT * FROM emp;

BEGIN

OPEN c_emp;

LOOP

FETCH c_emp INTO r_emp;

EXIT WHEN c_emp%NOTFOUND;

DBMS_OUT.PUT.PUT_LINE('Salary of Employee'||r_emp.ename||'is'|| r_emp.salary);

END LOOP;

CLOSE c_emp;

END;

%ROWTYPE也可以用游标名来定义,这样的话就必须要首先声明游标:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

SET SERVERIUTPUT ON

DECLARE

CURSOR c_emp IS SELECT ename,salary FROM emp;

R_emp c_emp%ROWTYPE;

BEGIN

OPEN c_emp;

LOOP

FETCH c_emp INTO r_emp;

EXIT WHEN c_emp%NOTFOUND;

DBMS_OUT.PUT.PUT_LINE('Salary of Employee'||r_emp.ename||'is'|| r_emp.salary);

END LOOP;

CLOSE c_emp;

END;

带参数的游标

与存储过程和函数相似,可以将参数传递给游标并在查询中使用。这对于处理在某种条件下打开游标的情
况非常有用。它的语法如下:

CURSOR cursor_name[(parameter[,parameter],...)] IS select_statement;

定义参数的语法如下:

Parameter_name [IN] data_type[{:=|DEFAULT} value]

与存储过程不同的是,游标只能接受传递的值,而不能返回值。参数只定义数据类型,没有大小。

另外可以给参数设定一个缺省值,当没有参数值传递给游标时,就使用缺省值。游标中定义的参数只是一个占位符,在别处引用该参数不一定可靠。

在打开游标时给参数赋值,语法如下:

OPEN cursor_name[value[,value]....];

参数值可以是文字或变量。

例:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

DECALRE

CURSOR c_dept IS SELECT * FROM dept ORDER BY deptno;

CURSOR c_emp (p_dept VARACHAR2) IS

SELECT ename,salary

FROM emp

WHERE deptno=p_dept

ORDER BY ename

r_dept DEPT%ROWTYPE;

v_ename EMP.ENAME%TYPE;

v_salary EMP.SALARY%TYPE;

v_tot_salary EMP.SALARY%TYPE;

BEGIN

OPEN c_dept;

LOOP

FETCH c_dept INTO r_dept;

EXIT WHEN c_dept%NOTFOUND;

DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname);

v_tot_salary:=0;

OPEN c_emp(r_dept.deptno);

LOOP

FETCH c_emp INTO v_ename,v_salary;

EXIT WHEN c_emp%NOTFOUND;

DBMS_OUTPUT.PUT_LINE('Name:'|| v_ename||' salary:'||v_salary);

v_tot_salary:=v_tot_salary+v_salary;

END LOOP;

CLOSE c_emp;

DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary);

END LOOP;

CLOSE c_dept;

END;

  游标FOR循环

在大多数时候我们在设计程序的时候都遵循下面的步骤:

1、打开游标
2、开始循环
3、从游标中取值
4、检查那一行被返回
5、处理
6、关闭循环
7、关闭游标

可以简单的把这一类代码称为游标用于循环。但还有一种循环与这种类型不相同,这就是FOR循环,用于
FOR循环的游标按照正常的声明方式声明,它的优点在于不需要显式的打开、关闭、取数据,测试数据的存在、定义存放数据的变量等等。

游标FOR循环的语法如下:

?

1

2

3

4

5

6

FOR record_name IN

(corsor_name[(parameter[,parameter]...)]

| (query_difinition)

LOOP

statements

END LOOP;

  下面我们用for循环重写上面的例子:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

  DECALRE

  CURSOR c_dept IS SELECT deptno,dname FROM dept ORDER BY deptno;

  CURSOR c_emp (p_dept VARACHAR2) IS

  SELECT ename,salary

  FROM emp

  WHERE deptno=p_dept

  ORDER BY ename

  v_tot_salary EMP.SALARY%TYPE;

BEGIN

  FOR r_dept IN c_dept LOOP

  DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname);

  v_tot_salary:=0;

  FOR r_emp IN c_emp(r_dept.deptno) LOOP

  DBMS_OUTPUT.PUT_LINE('Name:' || v_ename || 'salary:' || v_salary);

  v_tot_salary:=v_tot_salary+v_salary;

  END LOOP;

  DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary);

  END LOOP;

  END;

  在游标FOR循环中使用查询

在游标FOR循环中可以定义查询,由于没有显式声明所以游标没有名字,记录名通过游标查询来定义。

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

DECALRE

v_tot_salary EMP.SALARY%TYPE;

BEGIN

FOR r_dept IN (SELECT deptno,dname FROM dept ORDER BY deptno) LOOP

DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname);

v_tot_salary:=0;

FOR r_emp IN (SELECT ename,salary

FROM emp

WHERE deptno=p_dept

ORDER BY ename) LOOP

DBMS_OUTPUT.PUT_LINE('Name:'|| v_ename||' salary:'||v_salary);

v_tot_salary:=v_tot_salary+v_salary;

END LOOP;

DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary);

END LOOP;

END;

  游标中的子查询

  语法如下:

?

1

2

3

4

CURSOR C1 IS SELECT * FROM emp

WHERE deptno NOT IN (SELECT deptno

FROM dept

WHERE dname!='ACCOUNTING');

  可以看出与SQL中的子查询没有什么区别。

  游标中的更新和删除

  在PL/SQL中依然可以使用UPDATE和DELETE语句更新或删除数据行。显式游标只有在需要获得多行数据的情
况下使用。PL/SQL提供了仅仅使  用游标就可以执行删除或更新记录的方法。
  UPDATE或DELETE语句中的WHERE CURRENT OF子串专门处理要执行UPDATE或DELETE操作的表中取出的最近的
数据。要使用这个方法,在声明游标  时必须使用FOR UPDATE子串,当对话使用FOR UPDATE子串打开一个游标时,所有返回集中的数据行都将处于行级(ROW-LEVEL)独占式锁定,其
  他对象只能查询这些数据行,不能进行UPDATE、DELETE或SELECT...FOR            UPDATE操作。

语法:

  FOR UPDATE [OF [schema.]table.column[,[schema.]table.column]..
  [nowait]

  在多表查询中,使用OF子句来锁定特定的表,如果忽略了OF子句,那么所有表中选择的数据行都将被锁定。
如果这些数据行已经被其他会话锁定,那么正常情况下ORACLE将等待,直到数据行解锁。

  在UPDATE和DELETE中使用WHERE CURRENT OF子串的语法如下:

  WHERE{CURRENT OF cursor_name|search_condition}

  例:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

DELCARE

CURSOR c1 IS SELECT empno,salary

FROM emp

WHERE comm IS NULL

FOR UPDATE OF comm;

v_comm NUMBER(10,2);

BEGIN

FOR r1 IN c1 LOOP

IF r1.salary<500 THEN

v_comm:=r1.salary*0.25;

ELSEIF r1.salary<1000 THEN

v_comm:=r1.salary*0.20;

ELSEIF r1.salary<3000 THEN

v_comm:=r1.salary*0.15;

ELSE

v_comm:=r1.salary*0.12;

END IF;

UPDATE emp;

SET comm=v_comm

WHERE CURRENT OF c1l;

END LOOP;

END

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

-声明游标

--宗地表的调查日期LANDINFO_RESEARCHDATE

--复制到流程表的权属调查时间FLOW_REASEARCHTIME

DECLARE

cursor cur_sel_all is select LANDINFO_RESEARCHDATE,LANDINFO_LANDNO from t_leoa_landinfo; --定义游标

   l_date t_leoa_landinfo.landinfo_researchdate%type; --声明变量分别保存t_leoa_landinfo的各列

   l_landNo t_leoa_landinfo.landinfo_landno%type;

begin

open cur_sel_all;

  loop                 --循环取数,并将游标数据填充到返回纪录集合中

   fetch cur_sel_all into l_date,l_landNo;

   exit when cur_sel_all%NOTFOUND; --循环退出条件

   if cur_sel_all%FOUND then --获取数据

    update T_LEOA_BOOKFLOW t2 set FLOW_REASEARCHTIME = l_date where l_landNo = t2.landinfo_landno;

   end if;

  end loop;

close cur_sel_all;

end;

下面再分享一下另外一则游标使用方法的代码,具体如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

191

192

193

194

195

196

197

198

199

200

201

202

203

204

205

206

207

208

209

210

211

212

213

214

215

216

217

218

219

220

221

222

223

224

225

226

227

228

229

230

231

232

233

234

235

236

237

238

239

240

241

242

243

244

245

246

247

248

249

250

251

252

253

254

255

256

257

258

259

260

261

262

263

264

265

266

267

268

269

270

271

272

273

274

275

276

277

278

279

280

281

282

283

284

285

286

287

288

289

290

291

292

293

294

295

296

297

298

299

300

301

302

303

304

305

306

307

308

309

310

311

312

313

314

315

316

317

318

319

320

321

322

323

324

325

326

327

328

329

330

331

332

333

334

335

336

337

338

339

340

341

342

343

-- 声明游标;CURSOR cursor_name IS select_statement

--For 循环游标

--(1)定义游标

--(2)定义游标变量

--(3)使用for循环来使用这个游标

declare

    --类型定义

    cursor c_job

    is

    select empno,ename,job,sal

    from emp

    where job='MANAGER';

    --定义一个游标变量v_cinfo c_emp%ROWTYPE ,该类型为游标c_emp中的一行数据类型

    c_row c_job%rowtype;

begin

    for c_row in c_job loop

     dbms_output.put_line(c_row.empno||'-'||c_row.ename||'-'||c_row.job||'-'||c_row.sal);

    end loop;

end;

--Fetch游标

--使用的时候必须要明确的打开和关闭

declare

    --类型定义

    cursor c_job

    is

    select empno,ename,job,sal

    from emp

    where job='MANAGER';

    --定义一个游标变量

    c_row c_job%rowtype;

begin

    open c_job;

     loop

      --提取一行数据到c_row

      fetch c_job into c_row;

      --判读是否提取到值,没取到值就退出

      --取到值c_job%notfound 是false

      --取不到值c_job%notfound 是true

      exit when c_job%notfound;

      dbms_output.put_line(c_row.empno||'-'||c_row.ename||'-'||c_row.job||'-'||c_row.sal);

     end loop;

    --关闭游标

   close c_job;

end;

--1:任意执行一个update操作,用隐式游标sql的属性%found,%notfound,%rowcount,%isopen观察update语句的执行情况。

    begin

     update emp set ENAME='ALEARK' WHERE EMPNO=7469;

     if sql%isopen then

      dbms_output.put_line('Openging');

      else

       dbms_output.put_line('closing');

       end if;

     if sql%found then

      dbms_output.put_line('游标指向了有效行');--判断游标是否指向有效行

      else

       dbms_output.put_line('Sorry');

       end if;

       if sql%notfound then

        dbms_output.put_line('Also Sorry');

        else

         dbms_output.put_line('Haha');

         end if;

          dbms_output.put_line(sql%rowcount);

          exception

           when no_data_found then

            dbms_output.put_line('Sorry No data');

            when too_many_rows then

             dbms_output.put_line('Too Many rows');

             end;

declare

    empNumber emp.EMPNO%TYPE;

    empName emp.ENAME%TYPE;

    begin

     if sql%isopen then

      dbms_output.put_line('Cursor is opinging');

      else

       dbms_output.put_line('Cursor is Close');

       end if;

       if sql%notfound then

        dbms_output.put_line('No Value');

        else

         dbms_output.put_line(empNumber);

         end if;

         dbms_output.put_line(sql%rowcount);

         dbms_output.put_line('-------------');

         select EMPNO,ENAME into empNumber,empName from emp where EMPNO=7499;

         dbms_output.put_line(sql%rowcount);

        if sql%isopen then

        dbms_output.put_line('Cursor is opinging');

        else

        dbms_output.put_line('Cursor is Closing');

        end if;

         if sql%notfound then

         dbms_output.put_line('No Value');

         else

         dbms_output.put_line(empNumber);

         end if;

         exception

          when no_data_found then

           dbms_output.put_line('No Value');

           when too_many_rows then

            dbms_output.put_line('too many rows');

            end;

--2,使用游标和loop循环来显示所有部门的名称

--游标声明

declare

    cursor csr_dept

    is

    --select语句

    select DNAME

    from Depth;

    --指定行指针,这句话应该是指定和csr_dept行类型相同的变量

    row_dept csr_dept%rowtype;

begin

    --for循环

    for row_dept in csr_dept loop

      dbms_output.put_line('部门名称:'||row_dept.DNAME);

    end loop;

end;

--3,使用游标和while循环来显示所有部门的的地理位置(用%found属性)

declare

    --游标声明

    cursor csr_TestWhile

    is

    --select语句

    select LOC

    from Depth;

    --指定行指针

    row_loc csr_TestWhile%rowtype;

begin

 --打开游标

    open csr_TestWhile;

    --给第一行喂数据

    fetch csr_TestWhile into row_loc;

    --测试是否有数据,并执行循环

     while csr_TestWhile%found loop

      dbms_output.put_line('部门地点:'||row_loc.LOC);

      --给下一行喂数据

      fetch csr_TestWhile into row_loc;

     end loop;

    close csr_TestWhile;

end;

select * from emp

    

--4,接收用户输入的部门编号,用for循环和游标,打印出此部门的所有雇员的所有信息(使用循环游标)

--CURSOR cursor_name[(parameter[,parameter],...)] IS select_statement;

--定义参数的语法如下:Parameter_name [IN] data_type[{:=|DEFAULT} value]

declare

   CURSOR

   c_dept(p_deptNo number)

   is

   select * from emp where emp.depno=p_deptNo;

   r_emp emp%rowtype;

begin

    for r_emp in c_dept(20) loop

      dbms_output.put_line('员工号:'||r_emp.EMPNO||'员工名:'||r_emp.ENAME||'工资:'||r_emp.SAL);

    end loop;

end;

select * from emp 

--5:向游标传递一个工种,显示此工种的所有雇员的所有信息(使用参数游标)

declare

    cursor

    c_job(p_job nvarchar2)

    is

    select * from emp where JOB=p_job;

    r_job emp%rowtype;

begin

    for r_job in c_job('CLERK') loop

      dbms_output.put_line('员工号'||r_job.EMPNO||' '||'员工姓名'||r_job.ENAME);

    end loop;

end;

SELECT * FROM EMP

--6:用更新游标来为雇员加佣金:(用if实现,创建一个与emp表一摸一样的emp1表,对emp1表进行修改操作),并将更新前后的数据输出出来

--http://zheng12tian.iteye.com/blog/815770

    create table emp1 as select * from emp;

declare

    cursor

    csr_Update

    is

    select * from emp1 for update OF SAL;

    empInfo csr_Update%rowtype;

    saleInfo emp1.SAL%TYPE;

begin

  FOR empInfo IN csr_Update LOOP

   IF empInfo.SAL<1500 THEN

    saleInfo:=empInfo.SAL*1.2;

    elsif empInfo.SAL<2000 THEN

    saleInfo:=empInfo.SAL*1.5;

    elsif empInfo.SAL<3000 THEN

    saleInfo:=empInfo.SAL*2;

   END IF;

   UPDATE emp1 SET SAL=saleInfo WHERE CURRENT OF csr_Update;

   END LOOP;

END;

--7:编写一个PL/SQL程序块,对名字以‘A'或‘S'开始的所有雇员按他们的基本薪水(sal)的10%给他们加薪(对emp1表进行修改操作)

declare

   cursor

   csr_AddSal

   is

   select * from emp1 where ENAME LIKE 'A%' OR ENAME LIKE 'S%' for update OF SAL;

   r_AddSal csr_AddSal%rowtype;

   saleInfo emp1.SAL%TYPE;

begin

   for r_AddSal in csr_AddSal loop

     dbms_output.put_line(r_AddSal.ENAME||'原来的工资:'||r_AddSal.SAL);

     saleInfo:=r_AddSal.SAL*1.1;

     UPDATE emp1 SET SAL=saleInfo WHERE CURRENT OF csr_AddSal;

   end loop;

end;

--8:编写一个PL/SQL程序块,对所有的salesman增加佣金(comm)500

declare

   cursor

     csr_AddComm(p_job nvarchar2)

   is

     select * from emp1 where  JOB=p_job FOR UPDATE OF COMM;

   r_AddComm emp1%rowtype;

   commInfo emp1.comm%type;

begin

  for r_AddComm in csr_AddComm('SALESMAN') LOOP

    commInfo:=r_AddComm.COMM+500;

     UPDATE EMP1 SET COMM=commInfo where CURRENT OF csr_AddComm;

  END LOOP;

END;

--9:编写一个PL/SQL程序块,以提升2个资格最老的职员为MANAGER(工作时间越长,资格越老)

--(提示:可以定义一个变量作为计数器控制游标只提取两条数据;也可以在声明游标的时候把雇员中资格最老的两个人查出来放到游标中。)

declare

  cursor crs_testComput

  is

  select * from emp1 order by HIREDATE asc;

  --计数器

  top_two number:=2;

  r_testComput crs_testComput%rowtype;

begin

  open crs_testComput;

    FETCH crs_testComput INTO r_testComput;

     while top_two>0 loop

       dbms_output.put_line('员工姓名:'||r_testComput.ENAME||' 工作时间:'||r_testComput.HIREDATE);

       --计速器减一

       top_two:=top_two-1;

       FETCH crs_testComput INTO r_testComput;

      end loop;

   close crs_testComput;

end;

--10:编写一个PL/SQL程序块,对所有雇员按他们的基本薪水(sal)的20%为他们加薪,

--如果增加的薪水大于300就取消加薪(对emp1表进行修改操作,并将更新前后的数据输出出来)

declare

  cursor

    crs_UpadateSal

  is

    select * from emp1 for update of SAL;

    r_UpdateSal crs_UpadateSal%rowtype;

    salAdd emp1.sal%type;

    salInfo emp1.sal%type;

begin

    for r_UpdateSal in crs_UpadateSal loop

      salAdd:= r_UpdateSal.SAL*0.2;

      if salAdd>300 then

       salInfo:=r_UpdateSal.SAL;

       dbms_output.put_line(r_UpdateSal.ENAME||': 加薪失败。'||'薪水维持在:'||r_UpdateSal.SAL);

       else

       salInfo:=r_UpdateSal.SAL+salAdd;

       dbms_output.put_line(r_UpdateSal.ENAME||': 加薪成功.'||'薪水变为:'||salInfo);

      end if;

      update emp1 set SAL=salInfo where current of crs_UpadateSal;

    end loop;

end;

--11:将每位员工工作了多少年零多少月零多少天输出出来 

--近似

 --CEIL(n)函数:取大于等于数值n的最小整数

 --FLOOR(n)函数:取小于等于数值n的最大整数

 --truc的用法 http://publish.it168.com/2005/1028/20051028034101.shtml

declare

 cursor

  crs_WorkDay

  is

  select ENAME,HIREDATE, trunc(months_between(sysdate, hiredate) / 12) AS SPANDYEARS,

    trunc(mod(months_between(sysdate, hiredate), 12)) AS months,

    trunc(mod(mod(sysdate - hiredate, 365), 12)) as days

  from emp1;

 r_WorkDay crs_WorkDay%rowtype;

begin

  for  r_WorkDay in crs_WorkDay loop

  dbms_output.put_line(r_WorkDay.ENAME||'已经工作了'||r_WorkDay.SPANDYEARS||'年,零'||r_WorkDay.months||'月,零'||r_WorkDay.days||'天');

  end loop;

end;

--12:输入部门编号,按照下列加薪比例执行(用CASE实现,创建一个emp1表,修改emp1表的数据),并将更新前后的数据输出出来

-- deptno raise(%)

-- 10   5%

-- 20   10%

-- 30   15%

-- 40   20%

-- 加薪比例以现有的sal为标准

--CASE expr WHEN comparison_expr THEN return_expr

--[, WHEN comparison_expr THEN return_expr]... [ELSE else_expr] END

declare

   cursor

     crs_caseTest

     is

     select * from emp1 for update of SAL;

     r_caseTest crs_caseTest%rowtype;

     salInfo emp1.sal%type;

   begin

     for r_caseTest in crs_caseTest loop

     case

      when r_caseTest.DEPNO=10

      THEN salInfo:=r_caseTest.SAL*1.05;

      when r_caseTest.DEPNO=20

      THEN salInfo:=r_caseTest.SAL*1.1;

      when r_caseTest.DEPNO=30

      THEN salInfo:=r_caseTest.SAL*1.15;

      when r_caseTest.DEPNO=40

      THEN salInfo:=r_caseTest.SAL*1.2;

     end case;

     update emp1 set SAL=salInfo where current of crs_caseTest;

    end loop;

end;

--13:对每位员工的薪水进行判断,如果该员工薪水高于其所在部门的平均薪水,则将其薪水减50元,输出更新前后的薪水,员工姓名,所在部门编号。

--AVG([distinct|all] expr) over (analytic_clause)

---作用:

--按照analytic_clause中的规则求分组平均值。

 --分析函数语法:

 --FUNCTION_NAME(<argument>,<argument>...)

 --OVER

 --(<Partition-Clause><Order-by-Clause><Windowing Clause>)

   --PARTITION子句

   --按照表达式分区(就是分组),如果省略了分区子句,则全部的结果集被看作是一个单一的组

   select * from emp1

DECLARE

   CURSOR

   crs_testAvg

   IS

   select EMPNO,ENAME,JOB,SAL,DEPNO,AVG(SAL) OVER (PARTITION BY DEPNO ) AS DEP_AVG

   FROM EMP1 for update of SAL;

   r_testAvg crs_testAvg%rowtype;

   salInfo emp1.sal%type;

   begin

   for r_testAvg in crs_testAvg loop

   if r_testAvg.SAL>r_testAvg.DEP_AVG then

   salInfo:=r_testAvg.SAL-50;

   end if;

   update emp1 set SAL=salInfo where current of crs_testAvg;

   end loop;

end;

总结

以上就是本文关于Oracle中游标Cursor基本用法详解的全部内容,希望对大家有所帮助,欢迎参阅:oracle数据库导入导出命令解析、ORACLE SQL语句优化技术要点解析、浅谈oracle中单引号转义等,有什么问题可以随时留言,感谢大家!

Oracle中游标Cursor基本用法详解相关推荐

  1. 创建emp表 oracle,Oracle中创建和管理表详解

    Oracle中创建和管理表详解 更新时间:2013年08月01日 15:44:16   作者: 以下是对Oracle中的创建和管理表进行了详细的分析介绍,需要的朋友可以过来参考下 SQL> /* ...

  2. Oracle中游标Cursor介绍

    转自:http://zohan.group.iteye.com/group/wiki/2278-cursor Oracle中游标Cursor介绍 一  概念 游标是SQL的一个内存工作区,由系统或用户 ...

  3. Oracle之外键(Foreign Key)用法详解(一)

    Oracle外键(Foreign Key)用法详解(一) 1.目标 演示如何在Oracle数据库中使用外键 2.什么是外键? 1)在Oracle数据库中,外键是用来实现参照完整性的方法之一.打个形象的 ...

  4. Oracle中序列(Sequence)详解

    Oracle中序列(Sequence)详解 一 序列定义 序列(SEQUENCE)是序列号生成器,可以为表中的行自动生成序列号,产生一组等间隔的数值(类型为数字).不占用磁盘空间,占用内存. 其主要用 ...

  5. escape mysql_MySQL中ESCAPE关键字的用法详解

    MySQL转义 转义即表示转义字符原来的语义,一个转义字符的目的是开始一个字符序列,使得转义字符开头的该字符序列具有不同于该字符序列单独出现时的语义. MySQL中,转义字符以"" ...

  6. csh for循环_shell中的for循环用法详解_linux shell

    这篇文章主要介绍了shell中的for循环用法详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧 for 命令: for i i ...

  7. conv2d的输入_pytorch1.0中torch.nn.Conv2d用法详解

    Conv2d的简单使用 torch 包 nn 中 Conv2d 的用法与 tensorflow 中类似,但不完全一样. 在 torch 中,Conv2d 有几个基本的参数,分别是 in_channel ...

  8. 给mysql 授权 命令_mysql中授权命令grant用法详解:

    mysql中授权命令grant用法详解: mysql中可以给你一个用户授予如select,insert,update,delete等其中的一个或者多个权限,主要使用grant命令,用法格式为: gra ...

  9. python中如何反解函数_PyTorch中反卷积的用法详解

    pytorch中的 2D 卷积层 和 2D 反卷积层 函数分别如下: class torch.nn.Conv2d(in_channels, out_channels, kernel_size, str ...

最新文章

  1. C# 2.0 的partial
  2. Yii2 操作不同的数据库
  3. left join 与left outer join的区别
  4. jmeter将响应结果由Unicode转码成中文展示
  5. 互为质数的勾股数c语言,c语言 寻求勾股数满足x2+y2=z2的正整数x,y和z称为一组勾股数(pythagorean...
  6. OCM备考 三. Managing Database Availability 之RMAN日常操作
  7. 非规范SQL的sharding-jdbc实践
  8. jqgrid ajax加载数据,如何通过ajax在jqgrid中加载数据(json)?
  9. spring boot 如何修改默认端口号和context path
  10. 无法远程到2008R2的解决方法
  11. FreeChart柱状图中如何取消柱子的倒影
  12. bzoj 4337 树的同构
  13. js实现word生成书签_javascript下用ActiveXObject控件替换word书签,将内容导
  14. uniapp使用企业微信SDK踩坑指南
  15. yolov3/yolov4/yolov5/yolov6/yolov7/lite/fastdet/efficientdet各系列模型开发、项目交付、组合改造创新之—桥梁基建隧道裂痕裂缝检测实战
  16. android 画布实现签名,Android实现屏幕手写签名
  17. 【初等数论】整除、公约数、同余与剩余系
  18. 【渝粤题库】广东开放大学 管理学基础 形成性考核
  19. mysql使用报错1142(42000)解决方法
  20. 微信公众号商城是否需要云服务器,云商城公众号绑定、微信支付配置文档

热门文章

  1. Python茅台抢购脚本的使用说明!!
  2. 倚天屠龙记决战光明顶java,新倚天屠龙记张无忌决战光明顶 一人从头打到尾动作满分...
  3. 蓝牙怎么实现传输的_原来手机蓝牙有6个用法!很多人只用过一个,剩下5个很神奇...
  4. 用于门牌号码检测的深度学习
  5. 如何保存联想锁屏壁纸?
  6. 真香!美团java一面二面HR面面经
  7. 我的世界2b2t服务器ip显示没有,我的世界2b2t服务器怎么进,我的世界2b2t服务器怎么进入技巧...
  8. HTML5 - Three.js 3D特效学习
  9. 赋能开发:捷码携手达内教育打造IT职业教育新生态
  10. windows下最简单实用的拼接合并视频方法