备注:要进行下面的相关操作练习,请使用oracle内置的帐号scott/tiger进行登录。下面示例中所用到的表主要是scott内置帐户提供的emp和dept两张表。
一、视图和索引的使用
1、视图
视图的语法(视图的创建必须要有dba的权限)
create view 视图名称 as 查询语句 with read only
案例
create view v_emp as select ename, job from emp
规范:视图的名称一般情况下,是由v_加上要查询的表的表名组成。
eg:视图的数据来源emp表和user表 -------> v_emp_user
—视图的作用?
—第一:视图可以屏蔽掉一些敏感字段。
—第二:保证总部和分部数据及时统一。
2、索引
索引的作用:提高查询效率。
通过索引提高查询效率,也是实际开发中,最常用的一种方式。
索引有两种方式:单列索引和复合索引。
---单列索引
---创建单列索引
create index idx_ename on emp(ename);
---单列索引触发规则,条件必须是索引列中的原始值。
---单行函数,模糊查询,都会影响索引的触发。
select * from emp where ename='SCOTT'
---复合索引
---创建复合索引
create index idx_enamejob on emp(ename, job);
---复合索引中第一列为优先检索列
---如果要触发复合索引,必须包含有优先检索列中的原始值。
select * from emp where ename='SCOTT' and job='xx';---触发复合索引
select * from emp where ename='SCOTT' or job='xx';---不触发索引
select * from emp where ename='SCOTT';---触发单列索引。
3、oracle中的变量的定义
语法
变量的定义: 变量名 变量类型(oracle中的常用的4种类型:number varchar2 data blob)
变量的赋值 : := value
案例
---赋值操作可以使用:=也可以使用into查询语句赋值
declare
i number(2) := 10;
s varchar2(10) := '小明';
ena emp.ename%type;---引用型变量
emprow emp%rowtype;---记录型变量
begin
dbms_output.put_line(i);
dbms_output.put_line(s);
select ename into ena from emp where empno = 7788;
dbms_output.put_line(ena);
select * into emprow from emp where empno = 7788;
dbms_output.put_line(emprow.ename || '的工作为:' || emprow.job);
end;
二、if判断、循环和游标的用法
1、if判断的使用
示例
---输入小于18的数字,输出未成年
---输入大于18小于40的数字,输出中年人
---输入大于40的数字,输出老年人
declare
i number(3) := ⅈ -- &ii 输入指令
begin
if i<18 then
dbms_output.put_line('未成年');
elsif i<40 then
dbms_output.put_line('中年人');
else
dbms_output.put_line('老年人');
end if;
end;
2、sql中三种循环的用法
declare
--定义各种变量
begin
--编写各种业务代码
end;
while循环、exist、for循环
---while循环
declare
i number(2) := 1;
begin
while i<11 loop
dbms_output.put_line(i);
i := i+1;
end loop;
end;
---exit循环
declare
i number(2) := 1;
begin
loop
exit when i>10;
dbms_output.put_line(i);
i := i+1;
end loop;
end;
---for循环
declare
begin
for i in 1..10 loop
dbms_output.put_line(i);
end loop;
end;
3、游标的用法(理解为集合)
游标----- 理解为java中的集合,在循环游标时,可以理解游标记录着集合中每一行数据的下标index
---游标:可以存放多个对象,多行记录。
---输出emp表中所有员工的姓名
declare
cursor c1 is select * from emp;
emprow emp%rowtype;
begin
open c1;
loop
fetch c1 into emprow;
exit when c1%notfound;
dbms_output.put_line(emprow.ename);
end loop;
close c1;
end;
-----给指定部门员工涨工资
declare
cursor c2(eno emp.deptno%type) -- 括号中的值 表示的是一个参数
is select empno from emp where deptno = eno;-- 创建一个游标,来保存emp表中的所有empno的值
en emp.empno%type; -- 定义一个变量叫en,类型为emp表中的empno字段的类型
begin
open c2(10); -- 打开 游标c2
loop --开始循环
fetch c2 into en; -- 将游标c2循环时的每一个值赋值给变量en
exit when c2%notfound; -- 当游标c2的值为notfound(循环到最后一行数据)跳出循环
update emp set sal=sal+100 where empno=en;-- 满足条件时,执行的sql语句
commit; -- 提交事务
end loop; -- 结束循环
close c2; -- 结束游标c2
end;
三、存储过程和存储函数的使用
1、存储过程作用
将一段独立逻辑预先实现先,提供给其他程序使用。(类似于java中的封装)
示例:创建一个“给指定员工涨工资的逻辑代码)
----给指定员工涨100块钱
create or replace procedure p1(eno emp.empno%type)
is
begin
update emp set sal=sal+100 where empno = eno;
commit;
end;
调用上面的”存储过程“
----测试p1
declare
begin
p1(7788);
end;
2、存储函数的使用
要求:存储函数的返回值 必须要被接收
存储函数都会有一个返回值
示例: or replace 语句 都会带上
create or replace function f_yearsal(eno emp.empno%type) return number
is(as)
s number(10);
begin
select sal*12+nvl(comm, 0) into s from emp where empno = eno;
return s;
end;
定义的函数的用法
----存储函数在调用的时候,返回值需要接收。
declare
s number(10);
begin
s := f_yearsal(7788);
dbms_output.put_line(s);
end;
3、存储过程中 in和out关键字的用法
存储过程out关键字使用示例
---out类型参数如何使用
---使用存储过程来算年薪
create or replace procedure p_yearsal(eno emp.empno%type, yearsal out number)
is
s number(10);
c emp.comm%type;
begin
select sal*12, nvl(comm, 0) into s, c from emp where empno = eno;
yearsal := s+c;
end;
示例的使用
declare
yearsal number(10);
begin
p_yearsal(7788, yearsal);
dbms_output.put_line(yearsal);
end;
存储过程中,什么时候使用 in ,什么时候使用 out?
如果把存储过程理解为java中的一自定义方法,in 表示方法中的代码需要使用到的 变量;
方法中赋值的变量如果要提供给调用者使用,则用 out
java示例
// sal表示员工 的月薪;comm表示员工的奖金
public void yearSal(long sal,long comm,long yearSal){
yearSal = sal * 12 + comm;
}
//上面的java代码,表示使用一个不带返回值的方法,给变量yearSal来赋值
小结:
无论是存储过程,还是存储函数,最终的作用:都是为了在数据库层面来实现一个独立的业务逻辑(实现过程也可以放在java端来实现)。
四、触发器的使用
1、触发器的介绍
触发器是自动触发的
触发器只能在数据的 增、删和改 触发
2、触发器的种类:语句级触发器和行级触发器
语句级触发器:
针对数据库中数据的增、删、改;不会使用到内置的变量 :old和:new
示例
reate or replace trigger t1 -- 创建一个名称为t1的触发器
after -- 指定触发器触发的状态(before和after)
insert -- 监听操作的类型(insert / update /delete)
on person --针对哪一张表的上述3种操作
-- sql中编写语句块的语法
declare
begin
dbms_output.put_line('一个新员工入职');
end;
触发器的触发
insert into person values (1, '小红');
commit;
行级触发器
会使用到两个内置的变量 :old 更新数据之前的目标字段的值 ; :new 更新数据时,传入的 目标字段的值
示例
---raise_application_error(-20001~-20999之间, '错误提示信息');
create or replace trigger t2
before
update
on emp
for each row
declare
begin
if :old.sal>:new.sal then
raise_application_error(-20001, '不能给员工降薪');
end if;
end;
----触发t2
select * from emp where empno = 7788;
update emp set sal=sal-1 where empno = 7788;
commit;
3、使用触发器实现主键自增
示例
----触发器实现主键自增。【行级触发器】
---分析:在用户做插入操作的之前,拿到即将插入的数据,
------给该数据中的主键列赋值。
create or replace trigger auid
before
insert
on person
for each row
declare
begin
select s_person.nextval into :new.pid from dual;
end;
--查询person表数据
select * from person;
---使用auid实现主键自增
insert into person (pname) values ('a');
commit;
--表中已经存在 主键为1的数据,但仍然可以成功插入下面的数据
--原因: :new 表示的插入操作的新的那条数据。
-- 在做insert操作时,:new指的就是要新增的这条数据。而触发器又是监听的”before“的状态,所以在新增这条数据时,:new.pid的值就会被触发器进行修改,所以触发器一旦被触发,添加的数据中,”1“就会被触发器修改。
insert into person values (1, 'b');
commit;
4、关于工具配置
1)plsql developer工具插入中文乱码
”我的电脑“右键属性----- 高级系统设置-----环境变量
在环境变量中,配置一个系统变量:
NLS_LANG = SIMPLIFIED CHINESE_CHINA.ZHS16GBK
2)关于plsql developer工具中,查看当前用户下创建的所有表
将 工具---- 浏览器
窗口中的 “All Objects” ---- 切换为 ”My Objectes"