Oracle全教程

①创建表空间创建用户以及用户授权

(1)创建表空间

--创建表工作空间

create tablespace itheima

--在服务器上指定空间

datafile 'c:\itheima.dbf'

--为他设置空间大小

size 100m

--进行空间自动填充

autoextend on

--每次进行扩展10m

next 10m;

在服务器上指定空间的时候后缀名必须为dbf

(2)删除表空间

drop tablespace itheima;

执行完成以后,再去xp系统里删除即可;

(3)创建用户

--创建用户

create user itheima

--这是密码

identified by itheima

default tablespace itheima;

(4)用户授权

这是oracle数据库中常见的三种角色

给用户授权

oracle数据库中常见三种角色

connect 连接角色,基本角色

resource 开发者角色

dba 超级管理员角色

grant dba to itheima;--把dba权限授权给itheima

然后我们还需要切换到itheima角色

session→log off →all

然后在重新输入用户名以及密码进行登录即可

②数据类型介绍和表的创建

(1)数据类型介绍

oracle 数据类型

Varchar varchar2  表示一个字符串

NUMBER NUMBER(n) 表示一个整数,长度是N

NUMBER(m,n)表示一个小数 整数长度是m,小数点后边长度是n

data 表示日期类型

CLOB 表示大文本数据类型 可存4G 一般多用于存储文档

BLOB 表示二进制数据 可存4G 一般多用于存储视频文件

(2)创建表

创建表

create table person(

   pid number(20) ,

   pname varchar2(10)  

);

③修改表结构

--修改表结构

--添加一列

alter table person add (gender number(1));

--修改列类型

alter table person modify gender char(1);

--修改列名称

alter table person rename column gender to sex;

--删除一列

alter table person drop column sex;

④数据的增删改

--对表数据进行增删改

--对表中数据进行查找

select * from person ;

--往表中添加一条数据 ,只要是涉及到增删改的操作,都要进行提交,否则就是脏数据(所谓脏数据,就是没有提交的数据);

insert into person (pid,pname) values (1,'詹姆斯');

commit;

--对表中的数据进行修改

update person set pname='韦德' where pid =1;

commit;

--对表中的数据进行删除操作

三个删除

删除表中全部记录

delete from person;

删除表结构

drop table person;

先删除表,再创建表,效果等同于删除表中全部记录

在数据量大的情况下,尤其是表中带有索引的情况下,该操作效率极高

索引可以提高查询效率,但是会影响增删改效率;

truncate table person;

⑤序列的使用

序列

序列不真的属于任何一张表,但是可以逻辑和表做绑定

序列:默认从1开始,依次递增,主要用来给主键赋值使用

dual: 虚表, 只是为了补全语法,没有任何实际意义

create sequence s_person;

select s_person.nextval from dual;

添加一条语句

insert into person (pid ,panme) values (s_person.nextval,'小明' );

commit;

⑥Scott用户介绍

Scott用户,密码tiger

解锁Scott用户

alter user scott account unlock;

解锁Scott用户的密码[此句也可以用来重置密码]

alter user scott identified by tiger;

切换到scott用户 同上面的一样session→log off

⑦单行函数

单行函数:作用于一行,返回一个值

字符函数

select upper('yes') from dual ;   YES;小写变大写

select lower('YES') from dual ;   yes;大写变小写

数值函数

select round(10.5,1) from dual;  这是四舍五入  后边的数字表示保留的小数点后的位数,可以是复数;

select trunc (10.5,1) from dual; 直接截取,不看后边的数字是否大于5;

select mod(10,3) from dual; 求余数;

日期函数

查询出emp表中所有员工距离现在入职多少天

select sysdate-e.hiredate from emp e;

算出明天此时时刻

select sysdate+1 from dual;

查出emp表中所有员工入职距离现在多少个月

select months_between (sysdate,e.hiredate) from emp e;

查出emp表中所有员工入职距离现在多少年

select months_between (sysdate,e.hiredate)/12 from emp e;

查出emp表中所有员工入职距离现在多少周

select (sysdate-e.hiredate)/7 from emp e;

转换函数

日期转字符串

select to_char(sysdate,'fm yyyy-mm-dd hh24:mi:ss') from dual; fm可以去掉前面的数字0,hh24代表小时是按照24进行来显示的

字符串转日期

select to_date('2018-12-21 6:32:50','fm yyyy-mm-dd hh24:mi:ss') from dual;

通用函数

算出emp表中所有员工的年薪

奖金里边有null值,如果null值和任意类型数字做运算,结果都是null;

select e.sal*12+nvl(e.comm,0) from emp e; nvl(e.comm,0)功能就是有结果就用前边的,如果为null,则用0替代null;

⑧条件表达式

条件表达式

给emp表中的员工起中文名字

mysql和oracle通用

select e.ename,

case e.ename

  when 'smith'then'曹操'

   when 'James'then'刘备'

     when 'Kobe'then'项羽'

       else '无名'

         end

from emp e;

判断emp表中的员工收入 大于3000显示为高收入,大于1500显示为中等收入,小于1500显示为低收入;

select e.sal,

case  

  when e.sal>3000 then'高收入'

   when e.sal>1500 then'中等收入'

       else '低收入'

         end

from emp e;

oracle专用条件表达式

oracle中除了起别名,都用单引号;

select e.ename

decode(e.ename,

'james' '曹操',

'aleen' '韦德',

'Kobe'  '科比'

'无名'

) "中文名"

from emp e;

⑨多行函数

多行函数:作用于多行,返回一个值

select count(1) from emp ; 查询总数量

select sum(sal) from emp;  查询工资总和

select max(sal) from emp; 查询最高工资

select min(sal) from emp; 查询最低工资

select avg(sal) from emp; 查询平均工资

10分组查询

分组查询

查询出每个部分的平均工资

分组查询中,出现在group by后边的原始列,才可以出现在select后边,没有出现在group by 后边的列,想要在select后边出现,必须加上聚合函数,

聚合函数有一个特性,可以把多行记录变成一个值;

select e.deptno,avg(e.sal)

from emp e

group by e.deptno;

查询出平均工资高于2000的部门信息

select e.deptno,avg(e.sal) asal

from emp e

group by e.deptno

having avg(e.sal)>2000;

所有条件都不能使用别名来进行判断

比如下面的条件语句也不能使用别名当条件

where优先级高于select 所以相当于先执行where但是你设置别名,表中根本没有所以根本不知道

select ename ,sal s from emp where sal>1500;

11多表查询中的一些概念

---多表查询中的一些概念

---笛卡尔积

select *

from emp e, dept d;

---等值连接

select *

from emp e, dept d

where e.deptno=d.deptno;

---内连接

select *

from emp e inner join dept d

on e.deptno = d.deptno;

---查询出所有部门,以及部门下的员工信息。【外连接】

select *

from emp e right join dept d

on e.deptno=d.deptno;

---查询所有员工信息,以及员工所属部门

select *

from emp e left join dept d

on e.deptno=d.deptno;

---oracle中专用外连接

select *

from emp e, dept d

where e.deptno(+) = d.deptno;

12自连接

select * from emp;

---查询出员工姓名,员工领导姓名

---自连接:自连接其实就是站在不同的角度把一张表看成多张表。

select e1.ename, e2.ename

from emp e1, emp e2

where e1.mgr = e2.empno;

------查询出员工姓名,员工部门名称,员工领导姓名,员工领导部门名称

select e1.ename, d1.dname, e2.ename, d2.dname

from emp e1, emp e2, dept d1, dept d2

where e1.mgr = e2.empno

and e1.deptno=d1.deptno

and e2.deptno=d2.deptno;

13子查询

---子查询

---子查询返回一个值

---查询出工资和SCOTT一样的员工信息

select * from emp where sal in

(select sal from emp where ename = 'SCOTT')

---子查询返回一个集合

---查询出工资和10号部门任意员工一样的员工信息

select * from emp where sal in

(select sal from emp where deptno = 10);

---子查询返回一张表

---查询出每个部门最低工资,和最低工资员工姓名,和该员工所在部门名称

---1,先查询出每个部门最低工资

select deptno, min(sal) msal

from emp

group by deptno;

---2,三表联查,得到最终结果。

select t.deptno, t.msal, e.ename, d.dname

from (select deptno, min(sal) msal

     from emp

     group by deptno) t, emp e, dept d

where t.deptno = e.deptno

and t.msal = e.sal

and e.deptno = d.deptno;

14分页查询

----oracle中的分页

---rownum行号:当我们做select操作的时候,

--每查询出一行记录,就会在该行上加上一个行号,

--行号从1开始,依次递增,不能跳着走。

----排序操作会影响rownum的顺序

select rownum, e.* from emp e order by e.sal desc

----如果涉及到排序,但是还要使用rownum的话,我们可以再次嵌套查询。

select rownum, t.* from(

select rownum, e.* from emp e order by e.sal desc) t;

----emp表工资倒叙排列后,每页五条记录,查询第二页。

----rownum行号不能写上大于一个正数。

select * from(

   select rownum rn, tt.* from(

         select * from emp order by sal desc

   ) tt where rownum<11

) where rn>5

15视图

---视图

---视图的概念:视图就是提供一个查询的窗口,所有数据来自于原表。

---查询语句创建表

create table emp as select * from scott.emp;

select * from emp;

---创建视图【必须有dba权限】

create view v_emp as select ename, job from emp;

---查询视图

select * from v_emp;

---修改视图[不推荐]

update v_emp set job='CLERK' where ename='ALLEN';

commit;

---创建只读视图

create view v_emp1 as select ename, job from emp with read only;

---视图的作用?

---第一:视图可以屏蔽掉一些敏感字段。

---第二:保证总部和分部数据及时统一。

16索引

---索引

--索引的概念:索引就是在表的列上构建一个二叉树

----达到大幅度提高查询效率的目的,但是索引会影响增删改的效率。

---单列索引

---创建单列索引

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';---触发单列索引。

17PLSQL编程语言定义变量

---pl/sql编程语言

---pl/sql编程语言是对sql语言的扩展,使得sql语言具有过程化编程的特性。

---pl/sql编程语言比一般的过程化编程语言,更加灵活高效。

---pl/sql编程语言主要用来编写存储过程和存储函数等。

---声明方法

---赋值操作可以使用:=也可以使用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;

18if判断

---pl/sql中的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;

19循环

---pl/sql中的loop循环

---用三种方式输出1到10是个数字

---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;

20游标

---游标:可以存放多个对象,多行记录。

---输出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;

  en emp.empno%type;

begin

  open c2(10);

    loop

       fetch c2 into en;

       exit when c2%notfound;

       update emp set sal=sal+100 where empno=en;

       commit;

    end loop;  

  close c2;

end;

----查询10号部门员工信息

select * from emp where deptno = 10;

21存储过程

---存储过程

--存储过程:存储过程就是提前已经编译好的一段pl/sql语言,放置在数据库端

--------可以直接被调用。这一段pl/sql一般都是固定步骤的业务。

----给指定员工涨100块钱

create or replace procedure p1(eno emp.empno%type)

is

begin

  update emp set sal=sal+100 where empno = eno;

  commit;

end;

select * from emp where empno = 7788;

----测试p1

declare

begin

  p1(7788);

end;

22存储函数

----通过存储函数实现计算指定员工的年薪

----存储过程和存储函数的参数都不能带长度

----存储函数的返回值类型不能带长度

create or replace function f_yearsal(eno emp.empno%type) return number

is

  s number(10);    

begin

  select sal*12+nvl(comm, 0) into s from emp where empno = eno;

  return s;

end;

----测试f_yearsal

----存储函数在调用的时候,返回值需要接收。

declare

  s number(10);

begin

  s := f_yearsal(7788);

  dbms_output.put_line(s);

end;

23out类型参数

--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;

---测试p_yearsal

declare

  yearsal number(10);

begin

  p_yearsal(7788, yearsal);

  dbms_output.put_line(yearsal);

end;

----in和out类型参数的区别是什么?

---凡是涉及到into查询语句赋值或者:=赋值操作的参数,都必须使用out来修饰。

24存储函数和存储过程的区别

---存储过程和存储函数的区别

---语法区别:关键字不一样,

------------存储函数比存储过程多了两个return。

---本质区别:存储函数有返回值,而存储过程没有返回值。

----------如果存储过程想实现有返回值的业务,我们就必须使用out类型的参数。

----------即便是存储过程使用了out类型的参数,起本质也不是真的有了返回值,

----------而是在存储过程内部给out类型参数赋值,在执行完毕后,我们直接拿到输出类型参数的值。

----我们可以使用存储函数有返回值的特性,来自定义函数。

----而存储过程不能用来自定义函数。

----案例需求:查询出员工姓名,员工所在部门名称。

----案例准备工作:把scott用户下的dept表复制到当前用户下。

create table dept as select * from scott.dept;

----使用传统方式来实现案例需求

select e.ename, d.dname

from emp e, dept d

where e.deptno=d.deptno;

----使用存储函数来实现提供一个部门编号,输出一个部门名称。

create or replace function fdna(dno dept.deptno%type) return dept.dname%type

is

  dna dept.dname%type;

begin

  select dname into dna from dept where deptno = dno;

  return dna;

end;

---使用fdna存储函数来实现案例需求:查询出员工姓名,员工所在部门名称。

select e.ename, fdna(e.deptno)

from emp e;

25触发器

---触发器,就是制定一个规则,在我们做增删改操作的时候,

----只要满足该规则,自动触发,无需调用。

----语句级触发器:不包含有for each row的触发器。

----行级触发器:包含有for each row的就是行级触发器。

-----------加for each row是为了使用:old或者:new对象或者一行记录。

---语句级触发器

----插入一条记录,输出一个新员工入职

create or replace trigger t1

after

insert

on person

declare

begin

  dbms_output.put_line('一个新员工入职');

end;

---触发t1

insert into person values (1, '小红');

commit;

select * from person;

---行级别触发器

---不能给员工降薪

---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;

26触发器实现主键自增

----触发器实现主键自增。【行级触发器】

---分析:在用户做插入操作的之前,拿到即将插入的数据,

------给该数据中的主键列赋值。

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;

insert into person values (1, 'b');

commit;

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343