PL/SQL程序设计

1 PL/SQL简介

什么是PL/SQL?

PL/SQL是 Procedure Language & Structured Query Language 的缩写。PL/SQL是对SQL语言存储过程语言的扩展。

指在SQL命令语言中增加了过程处理语句(如分支、循环等),使SQL语言具有过程处理能力。

把SQL语言的数据操纵能力与过程语言的数据处理能力结合起来,使得PLSQL面向过程但比过程语言简单、高效、灵活和实用。

2 PL/SQL的优点或特征

1 有利于客户/服务器环境应用的运行
对于客户/服务器环境来说,真正的瓶颈是网络上。无论网络多快,只要客户端与服务器进行大量的数据交换。应用运行的效率自然就回受到影响。如果使用PL/SQL进行编程,将这种具有大量数据处理的应用放在服务器端来执行。自然就省去了数据在网上的传输时间。
2 适合于客户环境
PL/SQL由于分为数据库PL/SQL部分和工具PL/SQL。对于客户端来说,PL/SQL可以嵌套到相应的工具中,客户端程序可以执行本地包含PL/SQL部分,也可以向服务发SQL命令或激活服务器端的PL/SQL程序运行。
3 过程化
PL/SQL是Oracle在标准SQL上的过程性扩展,不仅允许在PL/SQL程序内嵌入SQL语句,而且允许使用各种类型的条件分支语句和循环语句,可以多个应用程序之间共享其解决方案。
4 模块化
PL/SQL程序结构是一种描述性很强、界限分明的块结构、嵌套块结构,被分成单独的过程、函数、触发器,且可以把它们组合为程序包,提高程序的模块化能力。
5 运行错误的可处理性
使用PL/SQL提供的异常处理(EXCEPTION),开发人员可集中处理各种ORACLE错误和PL/SQL错误,或处理系统错误与自定义错误,以增强应用程序的健壮性。
6 提供大量内置程序包
ORACLE提供了大量的内置程序包。通过这些程序包能够实现DBS的一些低层操作、高级功能,不论对DBA还是应用开发人员都具有重要作用。

当然还有其它的一些优点如:更好的性能、可移植性和兼容性、可维护性、易用性与快速性等。

3 PL/SQL 可用的SQL语句

PL/SQL是ORACLE系统的核心语言,现在ORACLE的许多部件都是由PL/SQL写成。在PL/SQL中可以使用的SQL语句有:
INSERT,UPDATE,DELETE,SELECT INTO,COMMIT,ROLLBACK,SAVEPOINT。
提示:在 PL/SQL中只能用 SQL语句中的 DML 部分,不能用 DDL 部分,如果要在PL/SQL中使用DDL(如CREATE table 等)的话,只能以动态的方式来使用。
ORACLE 的 PL/SQL 组件在对 PL/SQL 程序进行解释时,同时对在其所使用的表名、列名及数据类型进行检查
PL/SQL 可以在SQL*PLUS 中使用。
PL/SQL 可以在高级语言中使用。
PL/SQL可以在ORACLE的开发工具中使用(如:SQL Developer或Procedure Builder等)。
其它开发工具也可以调用PL/SQL编写的过程和函数,如Power Builder 等都可以调用服务器端的PL/SQL过程。

4 运行PL/SQL程序

5 PL/SQL程序结构及组成 

6 变量和常量说明

下面的所有练习使用的sqldeveloper工具。

常用快捷的alt+f10 打开一个SQL工作表。

sqldeveloper工具的安装及其使用:sqldeveloper安装及其使用教程

2 第一个PL/SQL程序

SQL> --如果要在屏幕上输出信息,需要将serveroutput开关打开
SQL> set serveroutput on
SQL> --第一个pl/sql程序,控制台输出hello world
SQL> declare
  2  begin
  3    dbms_output.put_line('hello world');
  4  end;
  5  /
hello world

PL/SQL procedure successfully completed.

3 引用型变量

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on

--引用型变量: 查询并打印7839的姓名和薪水

declare
  --定义变量保存姓名和薪水
  --pename varchar2(20);
  --psal   number;
  pename emp.ename%type;
  psal   emp.sal%type;
begin
  --得到7839的姓名和薪水
  select ename,sal into pename,psal from emp where empno=7839;

  --打印
  dbms_output.put_line(pename||'的薪水是'||psal);
end;
/

4 记录型变量

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on

--记录型变量: 查询并打印7839的姓名和薪水

declare
  --定义记录型变量:代表一行
  emp_rec emp%rowtype;
begin
  select * into emp_rec from emp where empno=7839;
  
  dbms_output.put_line(emp_rec.ename||'的薪水是'||emp_rec.sal);
end;

5 if语句

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on

-- 判断用户从键盘输入的数字

--接受键盘输入
--变量num:是一个地址值,在该地址上保存了输入的值
accept num prompt '请输入一个数字';

declare 
  --定义变量保存输入 的数字
  pnum number := #
begin
  if pnum = 0 then dbms_output.put_line('您输入的是0');
     elsif pnum = 1 then dbms_output.put_line('您输入的是1');
     elsif pnum = 2 then dbms_output.put_line('您输入的是2');
     else dbms_output.put_line('其他数字');
  end if;
end;
/

6 求两个数的和

----如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on

-- 例6:求两个数得和
-- 接收键盘输入
accept num1 prompt '请输入第一个数'
accept num2 prompt '请输入第二个数'
declare
  --定义变量保存输入得数字
  pnum1 number :=&num1;
  pnum2 number :=&num2;
  psum number :=0;
begin
  psum :=pnum1+pnum2;
  DBMS_OUTPUT.PUT_LINE(pnum1||'+'||pnum2||'='||psum);
end;
/

7 求1加到n的和使用while

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 使用while:求 1 加到 n 的和(循环)
-- 接收键盘输入
accept num prompt '请输入一个数'
declare
  --定义变量保存输入 的数字
  pnum number := #
  pn number :=1;
  psum number :=0;
begin
  WHILE pn<=pnum
  loop
    psum :=psum+pn;
    pn :=pn+1;
  end loop;
  -- 循环结束打印
  dbms_output.put_line('1加到'||pnum||'的和为'||psum);
end;
/

8 求1加到n的和使用do while

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 使用do while:求 1 加到 n 的和(循环)
-- 接收键盘输入
accept num prompt '请输入一个数'
declare
  --定义变量保存输入 的数字
  pnum number := &num;
  pn number :=1;
  psum number :=0;
begin
  loop
    exit when pn>pnum;
    psum :=psum+pn;
    pn :=pn+1;
  end loop;
  -- 循环结束打印
  dbms_output.put_line('1加到'||pnum||'的和为'||psum);
end;
/

9 求1加到n的和使用for

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 使用for:求 1 加到 n 的和(循环)
-- 接收键盘输入
accept num prompt '请输入一个数'
declare
  --定义变量保存输入 的数字
  pnum number := &num;
  psum number :=0;
begin
  for I in 1..pnum  --for循环的I直接使用,不用定义。1..100表示1到100的意思。不能有分号。
  loop
    psum :=psum+I;
  end loop;
  -- 循环结束打印
  dbms_output.put_line('1加到'||pnum||'的和为'||psum);
end;
/

10 给员工涨工资(游标)

知识点

代码

可以先把emp表复制一份

create table memp
as 
select * from emp;

执行下面的代码涨工资

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 给员工涨工资
-- 需求:按员工的工种长工资,总裁 1000 元,经理长 800 元,其他人员长 400 元。

declare 
  --定义光标
  cursor cemp is select empno,job from memp;
  pempno memp.empno%type;
  pjob   memp.job%type;
begin
  rollback;  
  --打开光标
  open cemp;  
  loop
       --取一个员工
       fetch cemp into pempno,pjob;
       exit when cemp%notfound;
       --判断职位
       if pjob = 'PRESIDENT' then update memp set sal=sal+1000 where empno=pempno;
          elsif pjob = 'MANAGER' then update memp set sal=sal+800 where empno=pempno;
          else update memp set sal=sal+400 where empno=pempno;
       end if;
  end loop;
  --关闭光标
  close cemp;
  --提交  ----> why?: 事务 ACID
  commit;
  dbms_output.put_line('完成');
end;
/

11 使用游标打印员工的姓名和薪水

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 使用游标打印员工的姓名和薪水
declare
  --定义光标(游标)
  cursor cemp is select ename,sal from emp;
  pename EMP.ENAME%type;
  psal emp.sal%type;
begin 
  --打开光标
  open cemp;
  loop
    --取当前记录
    fetch cemp into pename,psal;
    --exit when 没有取到记录退出
    exit when cemp%notfound;
    dbms_output.put_line(lower(pename)||'的薪水是'||to_char(psal,'L9,999.99'));
  end loop;
  --关闭光标
  close cemp;
end;
/

12 带参数的游标

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 带参数的游标  打印指定部门的员工的姓名和薪水
declare
  --定义光标(游标)
  cursor cemp(dno number) is select ename,sal from emp where deptno=dno;
  pename EMP.ENAME%type;
  psal emp.sal%type;
begin 
  --打开光标
  open cemp(10);
  loop
    --取当前记录
    fetch cemp into pename,psal;
    --exit when 没有取到记录退出
    exit when cemp%notfound;
    dbms_output.put_line(lower(pename)||'的薪水是'||to_char(psal,'L9,999.99'));
  end loop;
  --关闭光标
  close cemp;
end;
/

 

13 打印指定部门的员工信息

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 带参数的游标  打印指定部门的员工的姓名和薪水
--接收键盘输入
accept Jdeptno prompt '请输入要答应的部门号'

declare
  -- 接收键盘输入
  JPdeptno number :=&Jdeptno;
  --定义光标(游标)
  cursor cemp(dno number) is select ename,sal from emp where deptno=dno;
  pename EMP.ENAME%type;
  psal emp.sal%type;
begin 
  --打开光标
  open cemp(JPdeptno);
  loop
    --取当前记录
    fetch cemp into pename,psal;
    --exit when 没有取到记录退出
    exit when cemp%notfound;
    dbms_output.put_line(lower(pename)||'的薪水是'||to_char(psal,'L9,999.99'));
  end loop;
  --关闭光标
  close cemp;
end;
/

 

14 系统例外的使用

知识点

代码

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 系统列外  演示系统列为被0除
declare
  pnum number;
begin 
  pnum :=1/0;
  exception
    when zero_divide then dbms_output.put_line('1,0不能作为分母');
                          dbms_output.put_line('2,0不能作为分母');
                          dbms_output.put_line('3,0不能作为分母');
    when value_error then dbms_output.put_line('算术或者转换错误');
    when others then dbms_output.put_line('其它列外');
end;
/

15 用户自定义例外

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
-- 演示用户自定义例外 查询50号部门的员工(50号部门是不存在的)
/*自定义例外的三步曲:
  第一步:在declare节中定义例外。格式:例外名 exception
  第二步:在可行语句中引起例外   格式:raise 例外名
  第三步:在Exception节处理例外
*/
declare
  cursor cemp (dno number) is select ename from emp where deptno=dno;
  pename emp.ename%type;
  -- 自定义例外 员工不存在
  no_emp_found exception;
begin 
  --打开光标
  open cemp(50);
  --取出第一个员工
  fetch cemp into pename;
  if cemp%notfound then raise no_emp_found;
  end if;
  --关闭光标
  close cemp;
  
exception
  when no_emp_found then dbms_output.put_line('员工不存在或者部门不存在');
end;
/

16 统计每年入职的员工的个数

方法一:直接使用sql语句得到结果

 

或者下面的代码

 

方法二:使用PL/SQL程序

/*
1、SQL语句
select to_char(hiredate,'yyyy') from emp;
---> 集合 ---> 光标 ---> 循环---> 退出: notfound

2、变量:(*)初始值  (*)最终如何得到
每年入职的人数
count80 number := 0;
count81 number := 0;
count82 number := 0;
count87 number := 0;
*/
declare 
   --定义光标
   cursor cemp is select to_char(hiredate,'yyyy') from emp;
   phiredate varchar2(4);
   
   --每年入职的人数
  count80 number := 0;
  count81 number := 0;
  count82 number := 0;
  count87 number := 0;
begin
  --打开光标
  open cemp;
  loop
    --取一个员工的入职年份
    fetch cemp into phiredate;
    exit when cemp%notfound;
    
    --判断年份是哪一年
    if phiredate = '1980' then count80:=count80+1;
       elsif phiredate = '1981' then count81:=count81+1;
       elsif phiredate = '1982' then count82:=count82+1;
       else count87:=count87+1;
     end if;
  end loop;
  
  --关闭光标
  close cemp;
  
  --输出
  dbms_output.put_line('Total:'||(count80+count81+count82+count87));
  dbms_output.put_line('1980:'|| count80);
  dbms_output.put_line('1981:'|| count81);
  dbms_output.put_line('1982:'|| count82);
  dbms_output.put_line('1987:'|| count87);
end;
/

17 给员工涨工资

 

这里为了先不破坏emp表的数据。所以先创建一个表,复制emp的数据。

create table memp
as 
select * from emp;

实现涨工资的代码如下:

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
/*
1、SQL语句
selet empno,sal from emp order by sal;
---> 光标  ---> 循环  ---> 退出:1. 总额>5w   2. notfound

2、变量:(*)初始值  (*)最终如何得到
涨工资的人数: countEmp number := 0;
涨后的工资总额:salTotal number;
(1)select sum(sal) into salTotal from emp;
(2)涨后=涨前 + sal *0.1

练习: 人数:8    总额:50205.325
*/
declare
    cursor cemp is select empno,sal from memp order by sal;
    pempno memp.empno%type;
    psal   memp.sal%type;
    
    --涨工资的人数: 
    countEmp number := 0;
    --涨后的工资总额:
    salTotal number;
begin
    --得到工资总额的初始值
    select sum(sal) into salTotal from memp;
    
    open cemp;
    loop
         -- 1. 总额 >5w
         exit when salTotal > 50000;
         --取一个员工
         fetch cemp into pempno,psal;
         --2. notfound
         exit when cemp%notfound;
         --先判断,假设给员工涨工资后,工资总额是否超过5万,没有超过就涨,超过了就不涨
         exit when (salTotal+psal*0.1)>50000;
         
         --没有超过5万,给员工涨工资
         update memp set sal=sal*1.1 where empno=pempno;
         --人数+1
         countEmp := countEmp +1;
         --涨后=涨前 + sal *0.1
         salTotal := salTotal + psal * 0.1;

    end loop;
    close cemp;
    
    commit;
    dbms_output.put_line('人数:'||countEmp||'    总额:'||salTotal);
end;
/

 

18 按部门统计各工资段的职工人数

创建表的代码如下:

create table msg(
  pdeptno number,
  count1 number,
  count2 number,
  count3 number,
  saltotal number
);

代码实现:

/*
1、SQL语句
部门:select deptno from dept; ---> 光标
部门中员工的薪水: select sal from emp where deptno=?? ---> 带参数的光标

2、变量:(*)初始值  (*)最终如何得到
每个段的人数
count1 number; count2 number; count3 number;
部门的工资总额
salTotal number := 0;
(1)select sum(sal) into salTotal  from emp where deptno=??
(2)累加
*/
declare
  --部门
  cursor cdept is select deptno from dept;
  pdeptno dept.deptno%type;
  
  --部门中员工的薪水
  cursor cemp(dno number) is select sal from emp where deptno=dno;
  psal emp.sal%type;
  
  --每个段的人数
  count1 number; count2 number; count3 number;
  --部门的工资总额
  salTotal number := 0;
begin
  --部门
  open cdept;
  loop
       --取一个部门
       fetch cdept into pdeptno;
       exit when cdept%notfound;
       
       --初始化
       count1:=0; count2:=0; count3:=0;
       --得到部门的工资总额
       select sum(sal) into salTotal  from emp where deptno=pdeptno;
       
       --取部门的中员工薪水
       open cemp(pdeptno);
       loop
            --取一个员工的薪水
            fetch cemp into psal;
            exit when cemp%notfound;
            
            --判断
            if psal < 3000 then count1:=count1+1;
               elsif psal >=3000 and psal<6000 then count2:=count2+1;
               else count3:=count3+1;
            end if;
       end loop;
       close cemp;

       --保存结果
       insert into msg values(pdeptno,count1,count2,count3,nvl(saltotal,0));

  end loop;
  close cdept;
  
  commit;
  dbms_output.put_line('完成');
  
end;
/

执行完成后,查看msg表。

19 瀑布模型

20 存储过程入门

语法

 

代码一

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
create or replace procedure sayhelloworld
as 
--说明部分
begin
  dbms_output.put_line('hello world');
end sayhelloworld;
/

代码二

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
create or replace procedure sayhelloworld2
is
--说明部分
begin
  dbms_output.put_line('hello world 22222222');
end;
/

调用方式一

exec sayhelloworld2();

调用方式二

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
declare

begin
  sayhelloworld();
  sayhelloworld2();
end;
/

删除存储过程

drop procedure 过程名;

查看存储过程

双击可以看到源代码

21 带参数的存储过程

存储过程代码

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
--给指定的员工涨指定的工资,并且打印涨前和涨后的薪水
create or replace procedure raisesalary(
  eno in number,
  raisenum in number
)is
  --定义变量保存涨前的薪水
  psal emp.sal%type;
begin
  --获得涨前的薪水
  select sal into psal from memp where empno=eno;
  --涨钱
  update memp set sal = sal+raisenum where empno=eno;
  --要不要commit?不需要,让调用者决定什么时候提交
  dbms_output.put_line('涨前'||psal||'  涨后'||(psal+raisenum));
end;
/

调用存储过程

exec raisesalary(7839,-986);

22 存储函数

知识点

 

创建一个存储函数

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
--查询某个员工的年收入
create or replace function queryEmpIncome(eno in number) 
return number
is
       --定义变量保存月薪和奖金
       psal emp.sal%type;
       pcomm emp.comm%type;
begin
       --得到月薪和奖金
       select sal,comm into psal,pcomm from emp where empno=eno; 
       --返回年收入
       return psal*12+nvl(pcomm,0);
end queryEmpIncome;
/

调用上面的存储函数

--如果要在屏幕上输出信息,需要将serveroutput开关打开
set serveroutput on
declare
  annsal number;
begin
  annsal :=queryEmpIncome(7369);
  dbms_output.put_line('7369的年收入是'||annsal);
end;
/

存储函数的查看

点击可以打开源码

23 在java中调用存储过程

1.Java连接Oracle的jar包

2.常用配置

String driver = "oracle.jdbc.OracleDriver";
String url = "jdbc:oracle:thin:@192.168.47.10:1521/orcl";
String user = "scott";
String password = "tiger";  

3.准备一个存储过程

--查询某个员工的姓名 薪水和职位

/*
1、查询某个员工的所有信息 ---> out参数太多
2、查询某个部门中的所有员工信息 ----> 返回的是集合
*/

create or replace procedure queryEmpInformation(eno in number,
                                                pename out varchar2,
                                                psal   out number,
                                                pjob   out varchar2)
is
begin
  
   select ename,sal,job into pename,psal,pjob from emp where empno=eno;                                             

end queryEmpInformation;
/

4.1 编写一个JDBCUtils工具类

package utils;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;

public class JDBCUtils {

    private static String driver = "oracle.jdbc.OracleDriver";
    private static String url = "jdbc:oracle:thin:@192.168.47.10:1521/orcl";
    private static String user = "scott";
    private static String password = "tiger";
    
    static{
        //注册驱动
        //DriverManager.registerDriver(driver)
        try {
            Class.forName(driver);
        } catch (ClassNotFoundException e) {
            throw new ExceptionInInitializerError(e);
        }
    }
    public static Connection getConnection(){
        try {
            return DriverManager.getConnection(url, user, password);
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return null;
    }

    /*
     * 运行Java:
     * java -Xms100M -Xmx200M HelloWorld
     * 
     * 技术方向:
     * 1、性能优化
     * 2、故障诊断:死锁(JDK: ThreadDump)
     *               Oracle: 自动处理
     */
    public static void release(Connection conn,Statement st,ResultSet rs){
        if(rs != null){
            try {
                rs.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }finally{
                rs = null; ///-----> 原因:Java GC: Java的GC不受代码的控制 
                                      //java的垃圾回收不受代码的控制   gc()垃圾回收
            }
        }
        if(st != null){
            try {
                st.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }finally{
                st = null;
            }
        }
        if(conn != null){
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }finally{
                conn = null;
            }
        }
    }
}

4.2 测试存储过程

package utils;

import java.sql.CallableStatement;
import java.sql.Connection;
import java.sql.ResultSet;

import oracle.jdbc.OracleCallableStatement;
import oracle.jdbc.OracleTypes;

import org.junit.Test;

public class TestOracle {
/*
 * create or replace procedure queryEmpInformation(eno in number,
                                                pename out varchar2,
                                                psal   out number,
                                                pjob   out varchar2)
 */
    @Test
    public void testProcedure(){
        //{call <procedure-name>[(<arg1>,<arg2>, ...)]}
        String sql = "{call queryEmpInformation(?,?,?,?)}";
        
        Connection conn = null;
        CallableStatement call = null;
        try {
            conn = JDBCUtils.getConnection();
            call = conn.prepareCall(sql);
            
            //对于in参数,赋值
            call.setInt(1,7839);
            
            //对于out参数,申明
            call.registerOutParameter(2, OracleTypes.VARCHAR);
            call.registerOutParameter(3, OracleTypes.NUMBER);
            call.registerOutParameter(4, OracleTypes.VARCHAR);
            
            //执行
            call.execute();
            
            //输出
            String name = call.getString(2);
            double sal = call.getDouble(3);
            String job = call.getString(4);
            
            System.out.println(name+"\t"+sal+"\t"+job);
        } catch (Exception e) {
            e.printStackTrace();
        }finally{
            JDBCUtils.release(conn, call, null);
        }
    }
}

5.1  编写一个C3P0Utils工具类

 c3p0-config.xml配置文件内容如下

<?xml version="1.0" encoding="UTF-8"?>
<c3p0-config>

    <default-config>
        <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="jdbcUrl">jdbc:mysql:///student</property>
        <property name="user">root</property>
        <property name="password">root</property>
        <property name="initialPoolSize">5</property>
        <property name="maxPoolSize">20</property>
    </default-config>

    <named-config name="oracle">
        <property name="driverClass">oracle.jdbc.OracleDriver</property>
        <property name="jdbcUrl">jdbc:oracle:thin:@192.168.47.10:1521/orcl</property>
        <property name="user">scott</property>
        <property name="password">tiger</property>
    </named-config>
    
    <named-config name="mysql">
       <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="jdbcUrl">jdbc:mysql:///student</property>
        <property name="user">root</property>
        <property name="password">root</property>
    </named-config>
</c3p0-config>

C3P0Utils.java代码如下

package utils;

import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;

import javax.sql.DataSource;

import com.mchange.v2.c3p0.ComboPooledDataSource;

public class C3P0Utils {

    // 1 获得Connection ----- 从连接池中获取
    private static DataSource dataSource = new ComboPooledDataSource("oracle");

    // 2 创建ThreadLocal 存储的类型是Connection
    private static ThreadLocal<Connection> tl = new ThreadLocal<Connection>();

    // 3 直接可以获取一个连接池
    public static DataSource getDataSource() {
        return dataSource;
    }

    // 4 直接获取一个连接
    public static Connection getConnection() throws SQLException {
        return dataSource.getConnection();
    }

    // 5 获取绑定到ThreadLocal上的连接对象
    public static Connection getCurrentConnection() throws SQLException {
        //从ThreadLocal寻找 当前线程是否有对应Connection
        Connection con = tl.get();
        if (con == null) {
            //获得新的connection
            con = dataSource.getConnection();
            //将conn资源绑定到ThreadLocal(map)上
            tl.set(con);
        }
        return con;
    }

    // 6 开启事务
    public static void startTransaction() throws SQLException {
        Connection con = getCurrentConnection();
        if (con != null) {
            con.setAutoCommit(false);
        }
    }

    // 7 事务回滚
    public static void rollback() throws SQLException {
        Connection con = getCurrentConnection();
        if (con != null) {
            con.rollback();
        }
    }

    //  8 提交并且 关闭资源及从ThreadLocall中释放
    public static void commitAndRelease() throws SQLException {
        Connection con = getCurrentConnection();
        if (con != null) {
            con.commit(); // 事务提交
            con.close();// 关闭资源
            tl.remove();// 从线程绑定中移除
        }
    }

    // 9 关闭资源方法
    public static void release(Connection conn,Statement st,ResultSet rs){
        if(rs != null){
            try {
                rs.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }finally{
                rs = null; ///-----> 原因:Java GC: Java的GC不受代码的控制 
                                      //java的垃圾回收不受代码的控制   gc()垃圾回收
            }
        }
        if(st != null){
            try {
                st.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }finally{
                st = null;  //设置为空可以方便java的垃圾回收
            }
        }
        if(conn != null){
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }finally{
                conn = null;
            }
        }
    }
}

5.2  测试存储过程

/*
    create or replace procedure queryEmpInformation(eno in number,
            pename out varchar2,
            psal   out number,
            pjob   out varchar2)
    */
    @Test
    //测试在java中调用存储过程
    public void fun1() throws Exception{
        //{call <procedure-name>[(<arg1>,<arg2>, ...)]}
        String sql ="call queryEmpInformation(?,?,?,?)";
        Connection conn =null;
        CallableStatement call =null;
        try {
            conn = C3P0Utils.getConnection();
            call = conn.prepareCall(sql);
            //对于in参数,赋值
            call.setObject(1, 7839);
            
            //对于out参数,申明
            call.registerOutParameter(2, OracleTypes.VARCHAR);
            call.registerOutParameter(3, OracleTypes.NUMBER);
            call.registerOutParameter(4, OracleTypes.VARCHAR);
            
            
            //执行
            call.execute();
            
            //输出
            System.out.println(call.getObject(2)+"\t"+call.getObject(3)
            +"\t"+call.getObject(4));
            
            
        } catch (Exception e) {
        }finally{
            C3P0Utils.release(conn,call, null);
        }
        
    }

24 在java中调用存储函数

1 准备一个存储函数

create or replace function queryEmpIncome(eno in number) 
return number
is
       --定义变量保存月薪和奖金
       psal emp.sal%type;
       pcomm emp.comm%type;
begin
       --得到月薪和奖金
       select sal,comm into psal,pcomm from emp where empno=eno; 
       --返回年收入
       return psal*12+nvl(pcomm,0);
end queryEmpIncome;

2 测试

   /*
    * create or replace function queryEmpIncome(eno in number) 
    return number
    */
    @Test
    public void testFunction(){
        //{?= call <procedure-name>[(<arg1>,<arg2>, ...)]}
        String sql = "{?=call queryEmpIncome(?)}";
        
        Connection conn = null;
        CallableStatement call = null;
        try {
            conn = JDBCUtils.getConnection();
            call = conn.prepareCall(sql);
            
            call.registerOutParameter(1, OracleTypes.NUMBER);
            call.setInt(2, 7839);
            
            //执行
            call.execute();
            
            //取出年收入
            double income = call.getDouble(1);
            
            System.out.println(income);
        } catch (Exception e) {
            e.printStackTrace();
        }finally{
            JDBCUtils.release(conn, call, null);
        }        
    }

25 游标引用的java测试

1 包和体

--1、查询某个员工的所有信息--->out参数太多
--2、查询某个部门中的所有员工信息 ----> 返回的是集合

-- 申明包结构
create or replace package mypackage is
       type empcursor is ref cursor;
       procedure queryEmpList(dno in number,empList out empcursor);

end mypackage;
/
-- 创建包体
create or replace package body mypackage is

       procedure queryEmpList(dno in number,empList out empcursor)
       as
       begin
         
          open empList for select * from emp where deptno=dno;
       
       end;

end mypackage;
/

2 测试

@Test
    public void testCursor(){
        String sql = "{call mypackage.QUERYEMPLIST(?,?)}";
        Connection conn = null;
        CallableStatement call = null;
        ResultSet rs = null;
        try {
            conn = JDBCUtils.getConnection();
            call = conn.prepareCall(sql);
            //对于in参数,赋值
            call.setInt(1,20);
            //对于out参数,申明
            call.registerOutParameter(2, OracleTypes.CURSOR);
            //执行
            call.execute();
            //取出结果
            rs = ((OracleCallableStatement)call).getCursor(2);
            while(rs.next()){
                //取出一个员工
                String name = rs.getString("ename");
                double sal = rs.getDouble("sal");
                System.out.println(name+"\t"+sal);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }finally{
            JDBCUtils.release(conn, call, rs);
        }        
    }

3 注意事项

26 java操作Oracle数据库

1 导入驱动

创建 lib 目录, 用于存放当前项目需要的所有 jar

选择 jar 包, 右键执行 build path / Add to Build Path

2 测试查询

@Test
    // 测试查询
    public void fun1() throws Exception{
        //1 注册驱动
        Class.forName("oracle.jdbc.OracleDriver");
        //2 获取连接
        String url ="jdbc:oracle:thin:@192.168.47.10:1521/orcl";
        String user ="scott";
        String password ="tiger";
        Connection conn = DriverManager.getConnection(url, user, password);
        //3 获取执行者对象
        String sql ="select * from emp";
        PreparedStatement pst = conn.prepareStatement(sql);
        //4 执行
        ResultSet rs = pst.executeQuery();
        //5 结果处理
        while(rs.next()){
            System.out.println(rs.getObject("empno")+"\t"+
            rs.getObject("ename")+"\t"+rs.getObject("sal"));
        }
        //6 关闭资源
        rs.close();
        pst.close();
        conn.close();
    }

3 测试修改

@Test
    // 测试修改
    public void fun2() throws Exception{
        //1 注册驱动
        Class.forName("oracle.jdbc.OracleDriver");
        //2 获取连接
        String url ="jdbc:oracle:thin:@192.168.47.10:1521/orcl";
        String user ="scott";
        String password ="tiger";
        Connection conn = DriverManager.getConnection(url, user, password);
        //3 获取执行者对象
        String sql ="update emp set sal=sal-100 where empno=?";
        PreparedStatement pst = conn.prepareStatement(sql);
        //设置参数
        pst.setObject(1, 7369);
        
        //4 执行
        int rows = pst.executeUpdate();
        
        //5 结果处理
        System.out.println(rows);

        //6 关闭资源
        pst.close();
        conn.close();
    }

4 注意事项

27 第一个触发器

1 知识点

 

2 演示代码

--每当成功插入新员工后,自动打印“成功插入新员工”

create trigger firsttrigger
after insert
on emp
declare
begin
  dbms_output.put_line('成功插入新员工');
end;
/

 

28 触发器的应用-实施复杂的安全性检查

/*
实施复杂的安全性检查
禁止在非工作时间 插入新员工

1、周末:  to_char(sysdate,'day') in ('星期六','星期日')
2、上班前 下班后:to_number(to_char(sysdate,'hh24')) not between 9 and 17
*/
create or replace trigger securityemp
before insert
on emp
begin
   if to_char(sysdate,'day') in ('星期六','星期日') or 
      to_number(to_char(sysdate,'hh24')) not between 9 and 17 then
      --禁止insert
      raise_application_error(-20001,'禁止在非工作时间插入新员工');
   end if;
  
end securityemp;
/

29 触发器的应用-数据确认

/*
数据的确认
涨后的薪水不能少于涨前的薪水
*/
create or replace trigger checksalary
before update
on emp
for each row
begin
    if 涨后的薪水 < 涨前的薪水 then

       raise_application_error(-20002,'涨后的薪水不能少于涨前的薪水');
    end if;
end checksalary;
/

 

30 触发器的模拟MySQL的自增效果

1 创建表和序列

2 创建一个自增的触发器

3 测试

insert into test (name) values ('lisi');
insert into test (name) values ('zhangsan');
select * from test;

 

posted @ 2018-08-06 11:05  Jepson6669  阅读(893)  评论(0编辑  收藏  举报