一、实验目的
1. 对ORACLE数据库系统的用户权限管理有感性认识。
2. 事务是由用户定义的一个数据库的操作序列,这些操作要么全做,要么全不做,是一个不可分割的工作单位。 Oracle数据库使用事务机制来确保数据的一致性,数据库的事务机制是发生在第一个SQL语句执行时,结束于COMMIT或ROLLBACK命令执行时。只有一个事务执行完成后,另一个可执行的SQL语句才能执行。SQL执行时,所有的数据改变都是暂时的,只有结束事务的时候,才会真正写入数据库。通过本实验,对事务管理有感性认识。
二、实验原理
Oracle 权限设置
一、权限分类:
系统权限:
系统规定用户使用数据库的权限。(系统权限是对用户而言)。
实体权限:
某种权限用户对其它用户的表或视图的存取权限(针对表或视图而言)。
二、系统权限管理:
1、系统权限分类:
DBA:拥有全部特权,是系统最高权限,只有DBA才可以创建数据库结构。
RESOURCE:拥有Resource权限的用户只可以创建实体,不可以创建数据库结构。
CONNECT:拥有Connect权限的用户只可以登录Oracle,不可以创建实体,不可以创建数据库结构。
对于普通用户:授予connect, resource权限。
对于DBA管理用户:授予connect,resource, dba权限。
2、系统权限授权命令:
系统权限只能由DBA用户授出:sys, system(最开始只能是这两个用户)
授权命令:grant connect, resource, dba to 用户名1 [,用户名2]…;
普通用户通过授权可以具有与system相同的用户权限,但永远不能达到与sys用户相同的权限,system用户的权限也可以被回收。
例:以system用户身份连接数据库之后,执行:
create user cc identified by ccpassword ;
grant resource, connect, DBA to cc;
查询用户拥有哪里权限:
select * from dba_role_privs;
select * from dba_sys_privs;
select * from role_sys_privs;
删除用户: drop user 用户名 cascade;
//加上cascade则将用户连同其创建的东西全部删除
3、系统权限传递:
增加WITH ADMIN OPTION选项,则得到的权限可以传递。
grant connect, resource to CC with admin option; //可以传递所获权限
4、系统权限回收:系统权限只能由DBA用户回收
Revoke dba, resource from CC;
5、删除用户
drop user 用户名; //用户没有建任何实体
drop user 用户名 CASCADE; // 将用户及其所建实体全部删除
说明:当前正连接的用户不得删除。
三、实体权限管理
1、实体权限分类:
select, update, insert, alter, index, delete, all //all包括所有权限
execute //执行存储过程权限
user01:
grant select, update, insert on product to user02;
grant all on product to user02;
user02:
select * from user01.product;
// 此时user02查user_tables(连接中能显示的),不包括user01.product这个表,但如果查all_tables则可以查到,因为他可以访问。
2. 将表的操作权限授予全体用户:
grant all on product to public;
// public表示是所有的用户,这里的all权限不包括drop。
select owner, table_name from all_tables; // 用户可以查询的表
select table_name from user_tables; // 用户创建的表
select grantor, table_schema, table_name, privilege from all_tab_privs;
// 获权可以存取的表(被授权的)
select grantee, owner, table_name, privilege from user_tab_privs;
// 授出权限的表(授出的权限)
3. DBA用户可以操作全体用户的任意基表(无需授权,包括删除):
DBA用户:
Create table stud02.product(id number(10), name varchar2(20));
drop table stud02.emp;
create table stud02.employee as select * from scott.emp;
4. 实体权限传递(with grant option):
user01:
grant select, update on product to user02 with grant option;
// user02得到权限,并可以传递。
5. 实体权限回收:
user01:
Revoke select, update on product from user02; //传递的权限将全部丢失。
说明:如果取消某个用户的对象权限,那么对于这个用户使用WITH GRANT OPTION授予权限的用户来说,同样还会取消这些用户的相同权限,也就是说取消授权时级联的。
三、使用仪器、材料
Oracle 11g,windows10;
四、实验步骤
1、以SYSTEM身份连接到orcl数据库,创建新的用户并授权(如果原来已有这个用户,可以忽略本步骤):
2、以用户CC的身份建立连接,并在此连接下执行后面的操作;
select * from user_role_privs;
可以看到自己的权限,有多少种权限?
3、在CC连接中:拷贝代码运行,删去旧的同名数据表(如果是新创建的用户,此步骤可以省略):
4、在CC连接中:拷贝代码运行,建立表格及输入数据:
5、在CC连接中:确认orcl数据库中有这三个数据表,以及相应的数据
6、在CC连接中:查询用户CC的权限信息(每句单独执行):
7、在CC连接中:查询用户创建的表
8、在CC连接中:删去数据表BR,成功吗?
再次执行:
select table_name from user_tables;
显示什么结果?
9、回收用户CC的部分权限:以SYSTEM的身份连接(可以在SQL DEVELOPER的右上角切换),在此连接中执行
Revoke dba, resource from CC;
10、以CC的身份连接数据库(可以在SQL DEVELOPER的右上角选择连接),执行:Create Table Aa(cola int);
成功吗?为什么?
11、切换为SYSTEM的连接,执行切换为SYSTEM的连接,执行
grant resource to CC;
12、切换为CC的连接,执行
select * from user_role_privs;
可以看到自己的权限,有多少种权限?
Create Table Aa(cola int);
成功吗?为什么?
13、切换为SYSTEM的连接,执行
DROP USER CC;
成功吗?为什么?
14、删除连接CC,在SYSTEM的连接中,执行
DROP USER CC;
成功吗?为什么?
怎样才能成功删除用户CC ? 仅仅断开连接就行了吗?
为何要加Cascade ?删去后,还能以用户CC的身份连接吗?
15、在SYSTEM的连接中,重新创建用户CC和DD:查看这两个用户的权限
16、以CC和DD的身份分别建立连接,使得目前系统加上原来的SYSTEM共有三个连接(为识别方便,连接名和用户名一致)
选择CC连接,执行:
Create Table from_CC(内容 char(1));
Insert into from_cc values(‘a’);
select * from from_cc;
选择DD连接,执行:
select owner, table_name from all_tables where table_name=‘FROM_CC’;
显示什么?什么意思?
select * from from_cc;
显示什么?什么意思?
选择CC连接,执行:
grant all on from_cc to dd;
选择DD连接,执行:
select owner, table_name from all_tables where table_name=‘FROM_CC’;
显示什么?什么意思?
select * from from_cc;
显示什么?
select * from cc.from_cc;
显示什么?怎么回事?
刷新DD连接下的表,有没有“FROM_CC”这个表?
选择DD连接,执行:
Insert into cc.from_cc values(‘d’);
select * from cc.from_cc;
插入成功了吗?
17、选择CC连接,执行:
Revoke insert on FROM_CC from DD;
选择DD连接,执行:
select * from cc.from_cc;
能显示表的内容吗?
Insert into cc.from_cc values(‘y’);
select * from cc.from_cc;
插入成功了吗?
选择SYSTEM连接,执行:
Revoke dba, resource from DD;
选择DD连接,执行:
select * from cc.from_cc;
能显示表的内容吗?
Insert into cc.from_cc values(‘z’);
select * from cc.from_cc;
插入成功了吗?
选择CC连接,执行:
Revoke all on FROM_CC from DD;
select * from cc.from_cc;
选择DD连接,执行:
select * from cc.from_cc;
18、删除连接DD后,重新添加连接DD
select * from cc.from_cc;
select * from user_role_privs;
当前用户有多少种权限?
Create Table from_DD(哦 char(2));
能执行吗?
选择SYSTEM连接,执行:
grant resource to dd;
选择DD连接,执行:
Create Table from_DD(哦 char(2));
能执行吗?
删除连接DD后,重新添加连接DD
Create Table from_DD(哦 char(2));
Insert into from_dd values(‘甲’);
select * from from_dd;
能看到记录“甲”吗?
选择CC连接,执行:
select * from dd.from_dd;
能看到记录“甲”吗?
用户CC怎样才能看到from_dd表的所有记录?用户CC怎样才能为from_dd表添加记录?
19、创建新的用户并授权:
20、以用户CC的身份建立连接,并在此连接下执行后面的操作;
21、拷贝代码运行,建立表格及输入数据:
表*有多少行数据?
Insert Into Mytable Values(‘04’,‘钱老大’);
select * from mytable;
Insert Into Mytable Values(‘05’,‘钱老二’);
select * from mytable;
Rollback;
select * from mytable;
回退到哪里?表中剩下多少行数据?
Insert Into Mytable Values(‘01’,‘赵老大’);
select * from mytable;
Insert Into Mytable Values(‘02’,‘赵老二’);
select * from mytable;
Insert Into Mytable Values(‘03’,‘赵老三’);
select * from mytable;
commit;
Insert Into Mytable Values(‘04’,‘钱老大’);
select * from mytable;
Insert Into Mytable Values(‘05’,‘钱老二’);
select * from mytable;
Rollback;
select * from mytable;
回退到哪里?表中剩下多少行数据?commit的作用是什么?
Insert Into Mytable Values(‘04’,‘钱老大’);
Savepoint p1;
select * from mytable;
Insert Into Mytable Values(‘05’,‘钱老二’);
Savepoint p2;
select * from mytable;
Insert Into Mytable Values(‘06’,‘钱老三’);
select * from mytable;
Rollback to p2;
select * from mytable;
Rollback to p1;
select * from mytable;
运行结果显示,Savepoint的作用是什么?
五、实验结果及分析
1、以用户CC的身份建立连接,并在此连接下执行后面的操作;
select * from user_role_privs;
可以看到自己的权限,有多少种权限?
有三种权限,分别是CONNECT、DBA、RESOURCE
2、在CC连接中:拷贝代码运行,删去旧的同名数据表(如果是新创建的用户,此步骤可以省略):
Declare
tmp integer default 0;
Begin
select count() into tmp from user_tables where table_name=‘RB’;
if(tmp>0) then
execute immediate ‘drop table RB’;
end if;
select count() into tmp from user_tables where table_name=‘READER’;
if(tmp>0) then
execute immediate ‘drop table READER’;
end if;
select count(*) into tmp from user_tables where table_name=‘BOOK’;
if(tmp>0) then
execute immediate ‘drop table BOOK’;
end if;
end;
3、在CC连接中:拷贝代码运行,建立表格及输入数据:
create table Reader
(
RNO varchar2(4) primary key,
Rname varchar2(10) not null,
Rsex varchar2(2),
Rage integer,
Rboss varchar2(10),
Raddress varchar2(30)
) ;
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R001’,‘张三’,20,‘男’,‘李四’,‘416’);
create table Book
(
BNO varchar2(4),
Bname varchar2(50) not null,
Bauthor varchar2(50),
Bpress varchar2(50),
Bprice numeric(6,2),
primary key(BNO)
);
insert into book (BNO,Bauthor,Bname, bpress, bprice) values(‘B001’,‘严蔚敏’,‘数据结构’,‘清华大学出版社’,null);
create table RB
(
RNO varchar2(4),
BNO varchar2(4),
RBdate date default sysdate,
primary key(RNO,BNO),
foreign key (RNO) references Reader(RNO),
foreign key (BNO) references Book(BNO)
);
insert into RB (RNO,BNO) values (‘R001’,‘B001’);
4、在CC连接中:确认orcl数据库中有这三个数据表,以及相应的数据
select * from all_tables where owner=‘CC’;
依次执行下列语句:
select * from reader;
select * from book;
select * from rb;
5、在CC连接中:查询用户CC的权限信息(每句单独执行):
select * from dba_role_privs;
select * from dba_sys_privs;
select * from role_sys_privs;
第一条语句:
第二条语句:unlimited tablespace无限制的表空间权限
第三条语句:列出了很多角色的权限
6、在CC连接中:查询用户创建的表
select table_name from user_tables;
7、在CC连接中:删去数据表BR,成功吗?
成功了,显示表已删去
再次执行:
select table_name from user_tables;
显示什么结果?
查询结果只有reader表和book表了。
8、回收用户CC的部分权限:以SYSTEM的身份连接(可以在SQL DEVELOPER的右上角切换),在此连接中执行
Revoke dba, resource from CC;
再用select * from user_role_privs;查cc的权限,只剩下CONNECT了
9、以CC的身份连接数据库(可以在SQL DEVELOPER的右上角选择连接),执行:Create Table Aa(cola int);
成功吗?为什么?
执行失败,因为上一步已经回收了cc的DBA和resource权限,cc已经无权限创建表格了。
10、切换为SYSTEM的连接,执行切换为SYSTEM的连接,执行
grant resource to CC;
11、切换为CC的连接,执行
select * from user_role_privs;
可以看到自己的权限,有多少种权限?
有两种权限,CONNECT,RESOURCE
Create Table Aa(cola int);
成功吗?为什么?
成功了,因为system用户把resource权限授予了cc,所以cc有创建表的权限
12、切换为SYSTEM的连接,执行
DROP USER CC;
成功吗?为什么?
失败了,因为还没有断开连接。
13、删除连接CC,在SYSTEM的连接中,执行
DROP USER CC;
成功吗?为什么?
失败了,必须指定 CASCADE 以删除 ‘CC’
怎样才能成功删除用户CC ? 仅仅断开连接就行了吗?
仅仅断开连接不行,还需要指定CASCADE。
语句为 DROP USER CC CASCADE;
为何要加Cascade ?删去后,还能以用户CC的身份连接吗?
因为用户cc创建了一些基本表,在不先删除基本表的前提下想要删除用户cc,要cascade级联才能删除成功。
删去后,不能以cc身份连接。如图:
14、在SYSTEM的连接中,重新创建用户CC和DD:查看这两个用户的权限
Create User Cc Identified By Ccpassword;
Create User dd identified by ddpassword;
grant resource, connect, DBA to cc,dd;
查看这两个用户的权限(只有SYSTEM才有权执行下面这条语句):
select * from dba_role_privs where GRANTEE in (‘CC’,‘DD’);
15、以CC和DD的身份分别建立连接,使得目前系统加上原来的SYSTEM共有三个连接(为识别方便,连接名和用户名一致)
选择CC连接,执行:
Create Table from_CC(内容 char(1));
Insert into from_cc values(‘a’);
select * from from_cc;
选择DD连接,执行:
select owner, table_name from all_tables where table_name=‘FROM_CC’;
显示什么?什么意思?
查询表名为FROM_CC的表的创建者owner和表名table_name。
select * from from_cc;
显示什么?什么意思?
在用户dd中无法查询到表from_cc的数据信息。因为表from_cc在用户cc中创建的,且用户cc并未授权给用户dd。
选择CC连接,执行:
grant all on from_cc to dd;
选择DD连接,执行:
select owner, table_name from all_tables where table_name=‘FROM_CC’;
显示什么?什么意思?
查询表名为FROM_CC的表的创建者owner和表名table_name。
select * from from_cc;
显示什么?
select * from cc.from_cc;
显示什么?怎么回事?
虽然用户cc把对from_cc表的all操作权限授予用户dd了,但这个表终究还是
用户cc创建的,所以,想要在用户dd中查询from_cc表的数据,则要指明from_cc的创建用户。
刷新DD连接下的表,有没有“FROM_CC”这个表?
没有这个表,用户dd只是获得了对表from_cc的all权限,但其连接下的表并不会有from_cc。
选择DD连接,执行:
Insert into cc.from_cc values(‘d’);
select * from cc.from_cc;
插入成功了吗?
看起来是成功了,但去用户cc执行select * from from_cc;发现没有该数据
此处没有commit;提交插入的数据,所以在用户cc中查询表from_cc的数据时不会显示’d’的。
16、选择CC连接,执行:
Revoke insert on FROM_CC from DD;
选择DD连接,执行:
select * from cc.from_cc;
能显示表的内容吗?
可以。
Insert into cc.from_cc values(‘y’);
select * from cc.from_cc;
插入成功了吗?
看起来是成功了,实际上dd应该没有插入数据的权限
选择SYSTEM连接,执行:
Revoke dba, resource from DD;
选择DD连接,执行:
select * from cc.from_cc;
能显示表的内容吗?
可以显示。
Insert into cc.from_cc values(‘z’);
select * from cc.from_cc;
插入成功了吗?
看起来成功了,实际上dd没有插入数据的权限
选择CC连接,执行:
Revoke all on FROM_CC from DD;
select * from cc.from_cc;
选择DD连接,执行:
select * from cc.from_cc;
17、删除连接DD后,重新添加连接DD
select * from cc.from_cc;
显示表不存在
select * from user_role_privs;
当前用户有多少种权限?
只有一种权限,CONNECT
Create Table from_DD(哦 char(2));
能执行吗?
不能,dd现在没有建表的权限
选择SYSTEM连接,执行:
grant resource to dd;
选择DD连接,执行:
Create Table from_DD(哦 char(2));
能执行吗?
不能,仍然显示权限不足
删除连接DD后,重新添加连接DD
Create Table from_DD(哦 char(2));
Insert into from_dd values(‘甲’);
select * from from_dd;
能看到记录“甲”吗?
可以看到记录“甲”了
选择CC连接,执行:
select * from dd.from_dd;
能看到记录“甲”吗?
不能,只能看到一个空表
用户CC怎样才能看到from_dd表的所有记录?用户CC怎样才能为from_dd表添加记录?
在dd连接中执行语句grant all on from_dd to cc;授予cc所有权限
再在cc连接中执行语句select * from dd.from_dd;
这样就可以看到from__dd表的所有记录了。
在用户cc中执行语句:insert into dd.from_dd values(‘乙’);
再执行commit;就可以提交插入的数据,这样cc就可以为from__dd表添加数据了。
18、创建新的用户并授权:
create user cc1 identified by ccpassword ;
grant resource, connect, DBA to cc1;
19、以用户CC的身份建立连接,并在此连接下执行后面的操作;
create table mytable(tno char(2) primary key, tname char(8));
20、拷贝代码运行,建立表格及输入数据:
Insert Into Mytable Values(‘01’,‘赵老大’);
select * from mytable;
Insert Into Mytable Values(‘02’,‘赵老二’);
select * from mytable;
Insert Into Mytable Values(‘03’,‘赵老三’);
select * from mytable;
表*有多少行数据?
表*有三行数据
Insert Into Mytable Values(‘04’,‘钱老大’);
select * from mytable;
Insert Into Mytable Values(‘05’,‘钱老二’);
select * from mytable;
Rollback;
select * from mytable;
回退到哪里?表中剩下多少行数据?
回退到插入数据之前,表中已经没有数据了
Insert Into Mytable Values(‘01’,‘赵老大’);
select * from mytable;
Insert Into Mytable Values(‘02’,‘赵老二’);
select * from mytable;
Insert Into Mytable Values(‘03’,‘赵老三’);
select * from mytable;
commit;
Insert Into Mytable Values(‘04’,‘钱老大’);
select * from mytable;
Insert Into Mytable Values(‘05’,‘钱老二’);
select * from mytable;
Rollback;
select * from mytable;
回退到哪里?表中剩下多少行数据?commit的作用是什么?
回退到commit处,表中剩下三行数据,commit的作用是及时提交数据
Insert Into Mytable Values(‘04’,‘钱老大’);
Savepoint p1;
select * from mytable;
Insert Into Mytable Values(‘05’,‘钱老二’);
Savepoint p2;
select * from mytable;
Insert Into Mytable Values(‘06’,‘钱老三’);
select * from mytable;
Rollback to p2;
select * from mytable;
Rollback to p1;
select * from mytable;
运行结果显示,Savepoint的作用是什么?
保存点(savepoint)是事务过程中的一个逻辑点,用于取消部分事务,当结束事务时,会自动的删除该事务中所定义的所有保存点。当执行rollback时,通过指定保存点可以回退到指定的点。
六、实验心得体会
在实验中,我发现一些权限的授予和收回操作,需要断开目标用户的连接,再重连才能生效。通过这次实验,我掌握了数据库的权限控制方法,对语句间的逻辑也有了更深的认识,对数据库的认识也有了很大的进步。
实验2-2
Oracle数据库对象
一、实验目的
Oracle 数据库包含许多数据库对象,例如表、视图、索引、序列、存储过程、触发器等。表、视图、索引的操作在前面的实验中已经做了相应的练习,本实验将介绍如何使用序列、触发器和存储过程。同学们可以通过本实验掌握如何定义、使用删除这些数据库对象。
二、实验原理
PL/SQL 块结构说明
(1)声明部分 主要用于声明变量、常量、数据类型、游标、异常处理名称以及本地(局部)子程序定 义等。
(2)可执行部分 执行部分是 PL/SQL 块的功能实现部分。该部分通过变量赋值、流程控制、数据查询、 数据操纵、数据定义、事务控制、游标处理等实现块的功能。
(3)异常处理部分 异常处理部分用于处理该块执行过程中产生的异常。
三、使用仪器、材料
Oracle 19c,windows10;
四、实验内容
- 序列
a) 创建序列
b) 查看创建的序列对象
c) 使用序列
d) 修改序列
e) 删除序列 - 存储过程
a) 创建三个数据表
b) 插入数据创建存储过程
c) 创建存储过程,更新表中的数据
d) 执行存储过程,并比较存储过程执行前后的数据变化情况
e) 删除存储过程
f) 创建存储过程
g) 运行存储过程 - 触发器
a) 创建触发器
b) 创建触发器credit_id
c) 查看刚创建的触发器对象
d) 激活刚创建的触发器
五、实验结果及分析
1、 以SYSTEM连接数据库ORCL,执行以下语句查看对象:
select object_name,owner from all_objects where owner = ‘SYSTEM’;
显示有多少行?
共有461行。
2、 创建新的用户并授权:
create user cc identified by ccpassword ;
grant resource, connect, DBA to cc;
3、 以用户CC的身份建立连接,并在此连接下执行序列的操作:
select object_name,owner from all_objects where owner = ‘SYSTEM’;
显示有多少行?
有450行。
select object_name,owner from all_objects where owner = ‘CC’;
显示有多少行?
0行。
create sequence my_seq_01 increment by 1 start with 1 nomaxvalue nocycle;
create sequence my_seq_02 increment by 2 start with 1;
select object_name,owner from all_objects where owner = ‘CC’;
显示有多少行?
两行。
select object_name,Object_Type, owner from all_objects where owner = ‘SYSTEM’ and OBJECT_TYPE=‘SEQUENCE’;
显示有多少行?
7行。
Select Object_Name, Object_Type, Owner From All_Objects Where Owner = ‘CC’ and OBJECT_TYPE=‘SEQUENCE’;
显示有多少行?
两行。
select my_seq_01.nextval from dual;
重复执行上面的这条语句,得到什么序列?
结果是下图中的nextval列中数字从1开始,每执行一次语句就增加1。序列是1,2,3,4,5,6,7,8,9,10,11,12,…
alter sequence my_seq_01 increment by 10 ;
select my_seq_01.nextval from dual;
重复执行上面的这条语句,得到什么序列?
结果是下图中的nextval列中数字从上次的9开始,每执行一次语句就增加10。序列是19,29,39…
select my_seq_02.nextval from dual;
重复执行上面的这条语句,得到什么序列?
结果是下图中的nextval列中数字从1开始,每执行一次就增加2。序列是1,3,5,7,9…
drop sequence my_seq_02;
select my_seq_02.nextval from dual;
什么结果?
create sequence my_seq_02 increment by 3 start with 100;
select my_seq_02.nextval from dual;
重复执行上面的这条语句,得到什么序列?
结果是下图中的nextval列中数字从100开始,每执行一次就增加3。序列是100,103,106,109,112,115…
4、 在CC的连接中,执行存储过程的操作;
Declare
tmp integer default 0;
Begin
select count() into tmp from user_tables where table_name=‘SC’;
if(tmp>0) then
execute immediate 'drop table SC ';
end if;
select count() into tmp from user_tables where table_name=‘STUDENT’;
if(tmp>0) then
execute immediate 'drop table STUDENT ';
end if;
select count() into tmp from user_tables where table_name=‘COURSE’;
if(tmp>0) then
execute immediate 'drop table COURSE ';
end if;
Select count() into tmp From All_Objects Where OBJECT_NAME=‘SC_INS’ and OBJECT_TYPE=‘PROCEDURE’;
if(tmp>0) then
execute immediate 'drop PROCEDURE SC_INS ';
end if;
Select count(*) into tmp From All_Objects Where OBJECT_NAME=‘STUDENT_NO’ and OBJECT_TYPE=‘SEQUENCE’;
if(tmp>0) then
execute immediate 'drop SEQUENCE STUDENT_NO ';
end if;
end;
然后重新创建数据表:
create table student(sno int primary key, sname varchar(8));
create table course(cno int primary key, cname varchar(10));
create table sc(sno int, cno int, grade int,
primary key(sno,cno),
foreign key (sno) references student(sno), foreign key (cno) references course(cno));
成功了吗?上面的三个表的创建顺序能否任意调整?
成功了。前面两个表student和course的创建顺序可以任意调整,但sc表必须第三个创建,因为sc表的rno和cno属性参照的分别是student表的rno属性和course表的cno属性。
select object_name, Object_Type, owner from all_objects where owner = ‘CC’;
显示有多少行?索引是怎么产生的?
有8行。为了避免对基本表进行全表扫描,关系数据库管理系统一般都在主码上自动建立一个索引。
到SOL DEVELOPER的可视化窗口查看student表的索引信息:
查看SC表的索引,有哪些列来确定?依据是什么?索引名是什么?
由sno、cno列来确定,依据是主码,索引名是SYS_C007552
create sequence student_no increment by 1 start with 2012001;
insert into student values(student_no.nextval, ‘aaaaaa’);
insert into student values(student_no.nextval,‘bbbbbbb’);
insert into student values(student_no.nextval,‘ccccccc’);
insert into student values(student_no.nextval,‘ddddddd’);
commit;
select * from student;
查看student表的内容
insert into course values (105,‘程序设计’);
insert into course values (908,‘大学英语’);
insert into course values (433,‘数据结构’);
commit;
select * from course;
查看course表的内容
create procedure sc_ins(ino int,cno int,grade int) is
begin
if(grade>=0) then insert into sc values (ino,cno,grade);
else insert into sc values (ino,cno,null);
end if;
end;
成功了吗? 显示什么内容?
成功了,显示已编译
Select Object_Name, Object_Type, Owner From All_Objects Where Owner = ‘CC’ and OBJECT_TYPE=‘PROCEDURE’;
显示有多少行? 1行。
exec sc_ins (2012001,105,60);
exec sc_ins (2012001,908,0);
exec sc_ins (2012001,433,98);
exec sc_ins (2012002, 105,75);
exec sc_ins (2012002, 433,-1);
exec sc_ins (2012003, 105,64);
exec sc_ins (2012003, 908,90);
exec sc_ins (2012003, 433,-100);
成功了吗?
成功了。
select student.sno,sname,cname,grade
from student,course,sc
where student.sno=sc.sno and course.cno=sc.cno;
显示结果说明,成绩为负数的话,在数据表里是什么内容?存储过程有什么好处?
成绩为负数的话,在数据表里显示的是空值(null)。
好处:1、存储过程只在创造时进行编译,以后每次执行存储过程都不需再重新编译,而一般SQL语句每执行一次就编译一次,所以使用存储过程可提高数据库执行速度。
2、存储过程可以重复使用,可减少数据库开发人员的工作量。
分数为0,和分数为负数,有区别吗?
有区别。分数为0的时候,grade属性显示的是0,为负数时,显示的是null
5、 在CC的连接中,执行触发器的操作;
一、触发器只影响当前表:
alter table sc add (gradelevel char(1));
update sc set gradelevel=‘A’ where grade>=85;
update sc set gradelevel=‘B’ where grade>=75 AND grade<85;
update sc set gradelevel=‘C’ where grade>=60 AND grade<75;
update sc set gradelevel=‘D’ where grade<60;
select student.sno,sname,cname,grade,gradelevel
from student,course,sc
where student.sno=sc.sno and course.cno=sc.cno;
看到和上一步的显示结果有何区别?
增加了一列,根据分数划分成A、B、C、D四个等级。
create or replace trigger sc_ins before insert or update on sc
for each row
begin
if :new.grade>=85 then :new.gradelevel:=‘A’;
else if :new.grade>=75 then :new.gradelevel:=‘B’;
else if :new.grade>=60 then :new.gradelevel:=‘C’;
else if :new.grade>=60 then :new.gradelevel:=‘D’;
end if;
end if;
end if;
end if;
end;
执行成功了吗?
执行成功了,显示已编译。
select * from sc where sno=2012002;
insert into sc(sno,cno,grade) values (2012002,908,80);
select * from sc where sno=2012002;
刚刚插入的行,gradelevel列的值哪里来的?
是触发器的作用,前面的语句意思是插入或更新SC表每一行之前先创建或更新触发器trigger_ins,然后就是根据要插入的分数得到对应的A或B或C或D,即gradelevel列的值。然后再插入数据,查询数据时会发现gradelevel列的值就是分数对应的字母了。
二、触发器影响其他表:
alter table course add (maxgrade int);
update course set maxgrade=0;
select * from course;
表的结构改变成功了吗?修改成功没有?
表的结构改变了,修改成功了。
create or replace trigger course_ins before insert or update on sc
for each row
declare oldg int;
begin
select maxgrade into oldg from course where cno=:new.cno;
if oldg<:new.grade then update course set maxgrade=:new.grade where cno=:new.cno;
end if;
end course_ins;
触发器生成成功了吗?当前触发器的目录下有多少项?
生成成功了。触发器目录下有两项:COURESE_INS和SC_INS
select * from course;
记下当前各科的最高分数
所有最高分数都为0,因为上面步骤执行了update course set maxgrade=0;
把maxgrade属性列的值全部设置为0。
insert into sc(sno,cno,grade) values (2012004,908,99);
insert into sc(sno,cno,grade) values (2012004,433,88);
insert into sc(sno,cno,grade) values (2012004,105,59);
select * from sc;
select * from course;
2012004号同学的三科分数插入成功了吗?
当前各科的最高分有没有变化?
插入成功了,且最高分有了变化。
select * from sc where sno=2012003 and cno=105;
2012003号同学的105号课程的分数是多少?级别是什么?
分数是64分,级别为C。
update sc set grade=100 where sno=2012003 and cno=105;
select * from sc where sno=2012003 and cno=105;
select * from course;
2012003号同学的105号课程的分数修改成功了吗?级别(sc.gradelevel)有没有相应变化?当前各科的最高分(course.maxgrade)有没有变化?
一个修改语句,可以同时触发两个表的数据改变吗?
修改成功了,级别变化成A,程序设计的最高分变成了100。由上分析可得:一个修改语句可以同时出发两个表的数据改变。
六、实验心得和体会
通过这次实验,掌握数据库对象如触发器、存储过程和函数的定义和基本使用方法。
实验2-3
数据备份与恢复
一、实验目的
掌握ORACLE数据库系统的一种备份/恢复方法。
二、实验原理
Oracle数据库有三种标准的备份方法:
导出/导入
热备份(被备份的数据库不必关闭)
冷备份(被备份的数据库必须关闭)
导出备件是一种逻辑备份,冷备份和热备份是物理备份。
该实验做的属于逻辑备份。物理备份是指文件级的备份。
三、使用仪器、材料
Oracle 19c,windows10;
四、实验步骤
1、 以用户CC的身份建立连接,并在此连接下执行后面的操作。
2、 拷贝代码运行,删去旧的同名数据表。
3、 拷贝代码运行,建立表格及输入数据。
4、 以CC的身份连接,确认orcl数据库中有这三个数据表,以及相应的数据。
5、 打开“Database Configuration Assistant”,创建数据库mydb。
6、将数据库连接cc导出到D盘。
7、 确认D盘有文件“export from orcl.sql”。
8、 到ORACLE SQL DEVELOPER中以SYSTEM用户的身份建立新的连接(这个密码是前面自己新建数据库输入的密码)。
9、 在该连接下执行以下语句,即在数据库MYDB里增加用户CC,以便与备份的用户名一致(打开文件“export from orcl.sql”可见里面的语句带了用户名):
10、以用户CC的身份建立新的连接到数据库mydb:
11、 确认mydb中没有数据表READER、BOOK、RB:
12、在MYDB的CC用户连接中打开文件(或用记事本打开再复制粘贴)运行
13、确认MYDB中含有这三个表,且表的内容、约束等与数据库ORCL中的一致
14、为避免占用太多的系统资源,删去数据库MYDB。先删去有关MYDB的所有连接
15、打开“Database Configuration Assistant”
16、删去数据库MYDB:
注意:如果未删除数据库的连接,则无法删除数据库。
五、实验结果及分析
1、 以用户CC的身份建立连接,并在此连接下执行后面的操作;
2、 拷贝代码运行,删去旧的同名数据表:
Declare
tmp integer default 0;
Begin
select count() into tmp from user_tables where table_name=‘RB’;
if(tmp>0) then
execute immediate ‘drop table RB’;
end if;
select count() into tmp from user_tables where table_name=‘READER’;
if(tmp>0) then
execute immediate ‘drop table READER’;
end if;
select count(*) into tmp from user_tables where table_name=‘BOOK’;
if(tmp>0) then
execute immediate ‘drop table BOOK’;
end if;
end;
3、 拷贝代码运行,建立表格及输入数据:
create table Reader
(
RNO varchar2(4) primary key,
Rname varchar2(10) not null,
Rsex varchar2(2),
Rage integer,
Rboss varchar2(10),
Raddress varchar2(30)
) ;
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R001’,‘张三’,20,‘男’,‘李四’,‘416’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R002’,‘张三’,35,‘女’,null,‘417’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R003’,‘李四’,30,‘男’,null,‘416’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R004’,‘王五’,20,‘男’,null,‘417’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R005’,‘马六’,40,‘男’,null,‘416’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R006’,‘刘三’,20,‘男’,null,‘417’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R007’,‘王四’,40,‘男’,‘李四’,‘416’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R008’,‘李小龙’,20,‘男’,‘李四’,‘417’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R009’,‘王小倩’,40,‘男’,‘李四’,‘416’);
insert into Reader (RNO,Rname,Rage,Rsex,Rboss, Raddress) values(‘R010’,‘王一小’,20,‘男’,‘李四’,‘417’);
create table Book
(
BNO varchar2(4),
Bname varchar2(50) not null,
Bauthor varchar2(50),
Bpress varchar2(50),
Bprice numeric(6,2),
primary key(BNO)
);
insert into book (BNO,Bauthor,Bname, bpress, bprice) values(‘B001’,‘严蔚敏’,‘数据结构’,‘清华大学出版社’,null);
insert into book (BNO,Bauthor,Bname, bpress, bprice) values(‘B002’,‘唐发根’,‘数据结构’,‘北航出版社’,24);
insert into book (BNO,Bauthor,Bname, bpress, bprice) values(‘B003’,‘王珊’,‘数据库原理’,‘高等教育出版社’,40);
insert into book (BNO,Bauthor,Bname, bpress, bprice) values(‘B004’,‘张飞’,‘数据库原理’,‘清华大学出版社’,30);
insert into book (BNO,Bauthor,Bname, bpress, bprice) values(‘B005’,‘王珊’,‘数据库原理’,‘清华大学出版社’,null);
create table RB
(
RNO varchar2(4),
BNO varchar2(4),
RBdate date default sysdate,
primary key(RNO,BNO),
foreign key (RNO) references Reader(RNO),
foreign key (BNO) references Book(BNO)
);
insert into RB (RNO,BNO) values (‘R001’,‘B001’);
insert into RB (RNO,BNO) values (‘R001’,‘B002’);
insert into RB (RNO,BNO) values (‘R001’,‘B004’);
insert into RB (RNO,BNO) values (‘R002’,‘B001’);
insert into RB (RNO,BNO) values (‘R003’,‘B001’);
insert into RB (RNO,BNO) values (‘R004’,‘B001’);
insert into RB (RNO,BNO) values (‘R004’,‘B002’);
insert into RB (RNO,BNO) values (‘R005’,‘B001’);
insert into RB (RNO,BNO) values (‘R006’,‘B001’);
insert into RB (RNO,BNO) values (‘R006’,‘B003’);
insert into RB (RNO,BNO) values (‘R006’,‘B005’);
insert into RB (RNO,BNO) values (‘R006’,‘B002’);
insert into RB (RNO,BNO) values (‘R006’,‘B004’);
4、 以CC的身份连接,确认orcl数据库中有这三个数据表,以及相应的数据
select * from all_tables where owner=‘CC’ ;
select * from reader;
select * from book;
select * from rb;
5、 打开“Database Configuration Assistant”,创建数据库mydb
6、将数据库连接cc导出到D盘
7、 确认D盘有文件“export from orcl.sql”
export from orcl.sql 已在D盘中
8、 到ORACLE SQL DEVELOPER中以SYSTEM用户的身份建立新的连接(这个密码是前面自己新建数据库输入的密码):
9、 在该连接下执行以下语句,即在数据库MYDB里增加用户CC,以便与备份的用户名一致(打开文件“export from orcl.sql”可见里面的语句带了用户名):
create user cc identified by ccpassword ;
grant resource, connect, DBA to cc;
10、以用户CC的身份建立新的连接到数据库mydb:
11、 确认mydb中没有数据表READER、BOOK、RB:
12、在MYDB的CC用户连接中打开文件(或用记事本打开再复制粘贴)运行
13、确认MYDB中含有这三个表,且表的内容、约束等与数据库ORCL中的一致
mydb中含有这三个表,且表的内容、约束等与数据库ORCL中的一致
BOOK
READER
RB
14、为避免占用太多的系统资源,删去数据库MYDB。先删去有关MYDB的所有连接
删去 ccmydb 连接
删去mydbsys连接
15、打开“Database Configuration Assistant”
16、删去数据库MYDB:
注意:如果未删除数据库的连接,则无法删除数据库。
六、实验心得和体会
通过这次实验,对ORACLE数据库系统逻辑备份和恢复的方法有了初步的了解,更好的理解了数据库的知识,还了解到不能直接删除正在连接的用户,需要断开连接才能删除。数据库的备份和恢复是一项很重要的工作,还需多加练习使得操作更加熟练。
步骤繁琐,图片太多,我这里只有步骤和代码,请大家谅解一下