/*
      授权命令 grant 权限 to 用户
          授权可以授予多个权限
          grant  connect,resource to baidu   
      收回权限  revoke  权限 from  用户
          revoke dba from baidu
          
      创建用户 分配表空间--指定用户的默认表空间
         create  table p(..) 建表存在默认表空间
          --建表时候指定表空间 
         create table p(...) talebspace 表空间名
    */
    /*
     PlSql编程语言  procedure language 过程语言
          是在sql语句中加入一些处理过程的语句
          常见的条件表达式 if else  还有循环结构
     基本结构
        declare
           --声明部分 理解为定义
              --声明使用的变量
        begin
           --处理逻辑的代码块
        end;
    */
    --psSql简单示例
    declare
       v_n number := 1; --声明数值变量 赋值使用符号:=
       v_s varchar2(4) :='s'; --声明字符类型变量
       emp_ename emp.ename%type ;-- 引用类型变量
       emp_row   emp%rowtype ;-- 记录类型变量
    begin
       
       dbms_output.put_line('v_n====='||v_n); --输出语句相当于sys out 
       dbms_output.put_line('v_s====='||v_s);
       
       select ename into emp_ename from emp where empno=7499; --使用into关键字赋值
       dbms_output.put_line('emp_ename====='||emp_ename);
       
       select  * into  emp_row    from emp where empno = 7499; --赋值记录类型变量
       dbms_output.put_line('员工编号=='||emp_row.empno||'员工姓名'||emp_row.ename);
    end;
    /*
      plsql 的条件表达式判断
      if .. 处理语句 else if ..处理语句
      -------------------------
      if .. then  
      elsif .. then  
      else  
      end if;   
    */
    ---使用条件表达式判断员工的工资 使用汉字输出
    declare
      
       emp_row   emp%rowtype ;-- 记录类型变量
    begin
      
       select  * into  emp_row    from emp where empno = 7499; --赋值记录类型变量
       --使用表达式判断工资
       if emp_row.sal > 3000 then
          dbms_output.put_line('员工工资大于3000=='||emp_row.sal);
       elsif   emp_row.sal < 1000  then
          dbms_output.put_line('员工工资小于1000=='||emp_row.sal);
       else
           dbms_output.put_line('员工工资位于1000到3000之间=='||emp_row.sal);
       end if;
       
    end;

    /*
      循环结构
    第一种-----
      loop
        exit when 条件
      end loop;
    第二种 ---
      while 条件 loop
        
      end loop;
    第三种 ---
      for 变量  in  范围 loop
        
      end loop;
    */
    -------使用循环输出数字 1-----10
    /*
    第一种
      loop
        exit when 条件
      end loop;
    */
    declare
       v_n number :=1;
    begin
      
      loop 
        --只是用来判断退出使用的,并不是相当于if()else{}
        exit when v_n>10 ;   --退出条件
        dbms_output.put_line(v_n);
        v_n:=v_n+1;   --自增
      end loop;
    end;
    /*
    第二种
      while 条件  loop
        
      end loop;
    */
    declare
       v_n number :=1;
    begin
      
      while v_n<11  loop
        dbms_output.put_line(v_n);
        v_n:=v_n+1;   --自增
      end loop;
    end;
    /*
      第三种
      for 变量 in 范围 loop  变量的声明和范围的控制是由for循环自动执行
        
      end loop;
    */

    declare
    begin
      
      for i in 1..10  loop
        dbms_output.put_line(i);
      end loop;
    end;
    /*
      游标 光标  是用于接收查询的记录结果集 ResultSet 提示记录使用.next()
      游标的使用步骤
         声明游标 cursor 游标名 is select 语句 指定游标的记录结果集
         打开游标  open  游标名
         提取游标  fetch 游标名 into 记录类型变量 
         关闭游标  close cursor
      游标的两个属性  游标名%found     : 判断它有找到
                      游标名%notfound  : 判断它没有找到
        if  emp_cursor%found then
           dbms_output.put_line('found');
         elsif  emp_cursor%notfound then
           dbms_output.put_line('notfound');
         elsif emp_cursor%notfound is null then
            dbms_output.put_line('null');
         end if;
    */
    --使用while循环结构演示游标
    declare
         --声明游标
         cursor emp_cursor is select * from emp;
         --声明记录类型变量 用于接收游标提取的记录
         emp_row  emp%rowtype; 
    begin
         --打开游标
         open emp_cursor; 
         --提取游标(判断下一个是否有值)
         fetch emp_cursor into emp_row ;
         --有值就执行while循环
         while emp_cursor%found  loop
            dbms_output.put_line(emp_row.empno||'员工姓名'||emp_row.ename);
            --继续提取游标(并判断下一个是否有值)
            fetch emp_cursor into emp_row ;
         end loop;
         close emp_cursor;
    end;
    /*
     loop  
        exit when 游标提取不到
     end loop
    */
    declare
         --声明游标
         cursor emp_cursor is select * from emp;
         --声明记录类型变量 用于接收游标提取的记录
         emp_row  emp%rowtype; 
    begin
         --打开游标
         open emp_cursor; 
         loop
           fetch emp_cursor into emp_row;
           exit when emp_cursor%notfound;
             dbms_output.put_line(emp_row.empno||'员工姓名'||emp_row.ename);
         end loop;
         close emp_cursor;
    end;
    --使用游标提取某个部门的员工信息
    --声明带参数的游标信息
    declare
         --声明游标
         cursor emp_cursor(dno number) is select * from emp where deptno = dno ;
         --声明记录类型变量 用于接收游标提取的记录
         emp_row  emp%rowtype; 
    begin
         --打开游标 时候传入参数
         open emp_cursor(10); 
         loop
           fetch emp_cursor into emp_row;
           exit when emp_cursor%notfound;
             dbms_output.put_line(emp_row.empno||'员工姓名'||emp_row.ename);
         end loop;
         close emp_cursor;
    end;
    /*
      错误信息开发中的异常
      数据库中叫做 例外
            异常的分类 1.系统异常 系统定义好的异常
                           
                       2.自定义的异常
                         new 自定义类继承Exception 自定义传值(错误代码,提示信息)
                         使用场景
                            不满足某些特定业务场景,抛出自定义异常
            异常的处理
               java  try{}catche(IndexOutOfBoundException e){}catche(Exception e){}
               java  try{}catche(Exception e){} catche(IndexOutOfBoundException e){}--报错
               数据库可以捕捉处理异常
                     exception   关键字捕捉异常
                       when 异常类型 then 处理语句   判断异常类型 处理异常
                            
    */
    --异常的简单示例
    /*
      --除0的异常 除数为0
      --赋值错误
      
    */

    declare
      v_n number :=0;
      v_m number :=1;
      
    begin
      v_m:='s';  --将字符串赋值给数值变量
      v_m:= v_m/v_n;
    exception
      
      when zero_divide then 
        dbms_output.put_line('除数不能为0');
      when value_error then
        dbms_output.put_line('赋值有误');

    end;
    ---处理太多记录数异常
    declare
      
       emp_row   emp%rowtype ;-- 记录类型变量
    begin
      
       select  * into  emp_row  from emp ; --赋值记录类型
       
    exception
      
       when too_many_rows then
          dbms_output.put_line('太多记录数'); 
        when others then    --others是最大范围的异常 相当于java 的 Exception
         dbms_output.put_line('其他异常');
    end;
    /*
       需求 :使用游标查询部门下的员工信息
              如果部门下没有员工 报错提示
       需要自定义异常 
       变量名  exception     --声明自定义异常
              
    */
    declare
      
       cursor emp_cursor is select * from emp where deptno= 40;  --游标结果集不存在
        emp_row   emp%rowtype ;-- 记录类型变量
       no_dept_emp  exception ;  --声明自定义异常
    begin
       open emp_cursor;  --打开游标
        
           fetch emp_cursor into emp_row;
           if emp_cursor%notfound then
             --没有员工  抛出错误异常
             raise no_dept_emp;
           end if;
       close emp_cursor;
    exception
       when no_dept_emp then 
         dbms_output.put_line('部门下面没人,快招人吧');
    end;

    /*
     
      存储过程 是一段封装好的代码块,过程是编译好放在服务器提供开发人员调用
      
              封装的代码块意义:  提升开发效率  可以复用 谁用直接调用
                                  提升运行效率 一调用直接运行
       语法:create [or repalce]   procedure 过程名称(参数名 out|in  参数类型) 
             as|is
                 --声明变量的部分 
             begin
                 --处理过程语句代码块
             end;
       调用存储过程
          在begin和end之间使用 过程名传参调用
    */
    --存储过程的简单示例  使用存储过程给某个员工增加工资100
    create or replace procedure add_sal(eno in number )
    as
      emp_sal number :=0;
    begin
      select sal into emp_sal from emp where empno = eno ;
      dbms_output.put_line('涨工资之前是===='||emp_sal);
      update emp set sal=sal+100 where empno = eno;
      select sal into emp_sal from emp where empno = eno ;
      dbms_output.put_line('涨工资之后是===='||emp_sal);
      commit;
    end;
    --------调用存储过程
    declare
    begin
      add_sal(7499);
    end;
    /*
      使用存储过程统计某个员工的年薪,年薪需要返回输出打印
      in 类型输入参数可以 省略 默认就是输入参数
    */
    create or replace procedure count_sal(eno number,year_sal out number)
    as
    begin
      
        select sal*12+nvl(comm,0) into year_sal  from emp where empno=eno; --使用into赋值给输出参数

    end;
    ----调用存储过程计算年薪
    declare
      v_emp_sal number :=0;  
    begin
      count_sal(7499,v_emp_sal);
      dbms_output.put_line('年薪为=='||v_emp_sal);
    end;
    /*
      使用存储过程 查询出某个部门的员工信息
      某个部门应该接受一个in类型的输入参数
      查询到的部门员工多条记录返回应该使用结果集 
      声明游标 cursor 游标名 is select 语句指定结果集
      系统引用游标
       sys_refcursor 
       声明系统引用游标  变量名 sys_refcursor;  --不需要指定结果集
       打开游标      open 系统引用游标 for select 语句  --使用for关键字装入数据
    */
    create or replace procedure dept_emp(dno number,cusor_emp out sys_refcursor)
    as
    begin
      --根据传进来的部门编号给游标装入结果集数据
      open cusor_emp for select * from emp  where deptno = dno;   
    end;
    ----调用存储过程查询部门下的员工
    declare
       cursor_emp sys_refcursor;  --声明系统引用游标传参使用
       emp_row emp%rowtype ;--记录类型变量
    begin
      dept_emp(10,cursor_emp);
      --提取游标中的数据
      loop
        fetch cursor_emp into emp_row;
        exit when cursor_emp%notfound;
        dbms_output.put_line('编号'||emp_row.empno||'姓名'||emp_row.ename);
      end loop;
      close cursor_emp; 
    end;
    /*
      存储函数 是一段封装好的代码块,是编译好放在服务器提供开发人员调用
      
              封装的代码块意义:  提升开发效率  可以复用 谁用直接调用
                                  提升运行效率 一调用直接运行
                                  
       语法:create [or repalce]   function 函数名称(参数名 out|in  参数类型) return 数据类型
                                                    in 代表传入参数,out 代表传出参数                        
             as|is
                 --声明变量的部分 
             begin
                 --处理过程语句代码块
                 --return 变量
             end;
       调用存储函数
          在begin和end之间使用 函数名传参调用 函数必须使用变量接收 返回值
      
    */
    --使用存储函数统计某个员工的年薪
    create or replace function count_emp_sal(eno number,year_sal out number) return number
    as
      v_sal number :=0;
    begin
      
        select sal*12+nvl(comm,0) into year_sal  from emp where empno=eno; --使用into赋值给输出参数
        return v_sal;
    end;
    --不带out类型输出参数统计年薪
    create or replace function count_sal_noout(eno number) return number
    as
      v_sal number :=0;
    begin
      
        select sal*12+nvl(comm,0) into v_sal  from emp where empno=eno; --使用into赋值给输出参数
        return v_sal;
    end;
    --调用函数统计年薪
    declare
      emp_sal number:=0;
      total_sal number :=0;
    begin
       --total_sal := count_emp_sal(7499,emp_sal);
       total_sal := count_sal_noout(7499);
       dbms_output.put_line(emp_sal);--0
       dbms_output.put_line(total_sal); --统计后年薪
    end;
    /*
       存储函数和过程的区别
       
           1.创建的关键字  procedure   funciton
           2.创建函数 必须使用return 声明函数的返回变量数据类型
           3.在函数的方法体内 必须使用return 返回一个变量
           4.函数的调用 必须有变量接收返回值
           5.函数可以用在select 查询语句中  select emp.*,count_sal_noout(empno) from emp;
       
       存储函数和过程使用场景
           开发规范 java代码待用过程 过程是用来处理业务逻辑代码
           如果逻辑中需要用到一些功能性的封装,可以调用函数
           90%情况下 函数和过程通用 过程可以调用函数,函数同样可以调用过程
       
    */

    /*
      触发器 是一个监视器,监视对表中数据的操作
           如果对数据的操作满足触发器的执行条件,
           触发器会自动运行
      触发器语法:
          create or repalce trigger 触发器名称
          after|before   --触发器执行时机
          insert|update|delete  --监视的动作
          on 表名       --表级触发器
          declare
          begin          
          end;
       行级触发器                    insert        update        delete
           :new   动作之后的记录   要插入的记录   修改后的记录     空
           :old   动作之前的记录      空          原始的记录     原始的记录
           
    */
    --创建触发器监视表,如果表中有数据插入,输出一个欢迎语句
    create or replace trigger insert_trigger
    after
    insert
    on p
    declare
       
    begin
       dbms_output.put_line('欢迎加入!');
    end;
    ----插入数据测试效果
    insert into p values(1,'zs');
    commit;
    --插入数据不能在休息日插入数据
    --休息日 周六和周日
    /*
     raise_application_error(v1,v2) v1错误代码  v2是提示语句
                                      -20000 -20999 
    */
    create or replace trigger insert_no_work
    before
    insert
    on p
    declare
       v_day varchar2(10) ;
    begin
      --获取到当前星期
      select to_char(sysdate,'day') into v_day from dual;
      --判断星期是否在休息日
      if trim(v_day)  in ('saturday','sunday') then
      --如果休息 错误提示
         raise_application_error(-20001,'不能休息日插入数据');
      end if;
    end;
    ----插入数据测试效果
    insert into p values(1,'zs');
    commit;

    --使用触发器监视表中数据修改,不能做降低工资的操作
    create or replace trigger can_not_low
    before
    update
    on emp
    for each row --行级触发器
    declare
      
    begin
      --获取到原始记录的工资  --获取修改后的工资
      if :old.sal > :new.sal then
      --谈错误框提示
      raise_application_error(-20002,'不能降低工资');
      end if;
    end;
    --修改员工的工资测试触发器
    update emp set sal=sal-1 where empno=7499;

    /*
      触发器实际应用 
          使用触发器实现 插入数据的id 自增长 面试题
    **/

    create or replace trigger auto_increment_id
    before    
    insert
    on test_trigger
    for each row 
    declare 
    begin
      --补全将要插入记录的id
      --补全的id 是自增长的数值 如果没有提前创建序列,需要提前创建序列   --创建序列  create sequence order_sequence 
      select order_sequence.nextval into :new.pid from dual;
    end;

    insert into test_trigger(pname,phone) values('zs','1234566');
    commit;

    package baidu;

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

import org.junit.Test;

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

public class TestJdbc {

     String driverClass = "oracle.jdbc.driver.OracleDriver";
     String url ="jdbc:oracle:thin:@192.168.17.128:1521:orcl";
     String user= "baidu_03";
     String password = "baidu_03";
     /*
      *测试jdbc连接数据库
      *
      * */
     @Test
     public void querEmp(){
         try{
         //加载驱动
         Class.forName(driverClass);
         //获取链接
         Connection con = DriverManager.getConnection(url, user,password);
         //获取预编译的statement
         PreparedStatement pst= con.prepareStatement("select * from emp");
         //执行查询
         ResultSet rs =  pst.executeQuery();
         //处理结果
         while(rs.next()){
             System.out.println(rs.getInt(1)+"员工姓名"+rs.getString("ename"));
         }
         rs.close();
         con.close();
         //关闭连接
         }catch(Exception e){
             e.printStackTrace();
         }
         
     }
     /*存储过程的调用
      * {call <procedure-name>[(<arg1>,<arg2>, ...)]}
        add_sal(eno number,addsal number)
      * */
     @Test
     public void callAddSal(){
         try{
         //加载驱动
         Class.forName(driverClass);
         //获取链接
         Connection con = DriverManager.getConnection(url, user,password);
         //获取预编译的statement
         CallableStatement pst= con.prepareCall("{call add_sal(?,?)}");
         pst.setInt(1, 7499);
         pst.setInt(2, 1000);
         //执行查询
         pst.execute();
         con.close();
         //关闭连接
         }catch(Exception e){
             e.printStackTrace();
         }
         
     }
     
     /*存储过程的调用
      * {call <procedure-name>[(<arg1>,<arg2>, ...)]}
        count_yearsal(eno number,total_year_sal out number)
      * */
     @Test
     public void callCountSal(){
         try{
         //加载驱动
         Class.forName(driverClass);
         //获取链接
         Connection con = DriverManager.getConnection(url, user,password);
         //获取预编译的statement
         CallableStatement pst= con.prepareCall("{call count_yearsal(?,?)}");
         pst.setInt(1, 7499);
         //注册输出参数
         pst.registerOutParameter(2, OracleTypes.NUMBER);
         //执行查询
         pst.execute();
         int total =  pst.getInt(2);
         System.out.println(total);
         con.close();
         //关闭连接
         }catch(Exception e){
             e.printStackTrace();
         }
         
     }
     /*
      * pro_dept_emp(dno number,dept_emp out sys_refcursor)
      * */
     @Test
     public void callProEmp(){
         try{
         //加载驱动
         Class.forName(driverClass);
         //获取链接
         Connection con = DriverManager.getConnection(url, user,password);
         //获取预编译的statement
         CallableStatement pst= con.prepareCall("{call pro_dept_emp(?,?)}");
         pst.setInt(1, 10);
         //注册输出参数
         pst.registerOutParameter(2, OracleTypes.CURSOR);
         //执行查询
         pst.execute();
         OracleCallableStatement ocs = (OracleCallableStatement)pst;
         ResultSet rs =  ocs.getCursor(2);
         while(rs.next()){
             System.out.println(rs.getInt(1)+"员工姓名"+rs.getString("ename"));
         }
         rs.close();
         ocs.close();
         pst.close();
         con.close();
         //关闭连接
         }catch(Exception e){
             e.printStackTrace();
         }
         
     }
}

public void show4(){
         try {
             
             Class.forName(driverClass);
             Connection con = DriverManager.getConnection(url, user,password);
             CallableStatement pst= con.prepareCall("{?= call count_sal_noout(?)}");

             //给第二个参数赋值
             pst.setLong(2, 7499);
            // stat2.setLong(2, empno);

             //声明第一个参数的类型
             pst.registerOutParameter(1, OracleTypes.NUMBER);
             pst.execute();
             OracleCallableStatement ocs = (OracleCallableStatement)pst;
             NUMBER num = ocs.getNUMBER(1);
             System.out.println(num);
//             long i = pst.getLong(1);
//             System.out.println(i);
         
             con.close();
             
         } catch (Exception e) {
             e.printStackTrace();
         }
     }
/*

 

posted on 2019-09-30 11:22  小小一  阅读(400)  评论(0编辑  收藏  举报