时间:2021-07-01 10:21:17 帮助过:16人阅读
2.Oracle,DB2,Sybase,MS SQL ,Mysql
3.实体,属性,关系 1521
4.oracle的服务:监听服务(远程访问需要)和数据库服务(要启动)
5.oracle的配置文件:
listener.ora(监听器服务的配置文件)和
tnsname.ora(监听器服务访问的配置文件)连接数据库的主机字符串(自定义:主机字符串包括远程IP
和数据库名了)
6.测试远程连接
net manager(配置主机字符串和监听器)
找安装路径的快捷方式:右击net manager打开文件
plsql工具:字体设置和显示行号 关键字大写
7.oracle的物理文件
分为三类:数据文件(.dbf)、控制文件(.ctl)、日志文件(.log)
8.oracle系统表(数据字典)
9.表空间(对应一个物理的磁盘存储位置)
CREATE TABLESPACE j1706 DATAFILE 'E:\oracle\product\10.2.0\oradata\orcl\j1706.dbf' SIZE 20M AUTOEXTEND ON ; 用户 授权 表
10.sql语句
SQL语句主要可以划分为以下几类:
DDL(Data Definition Language):数据定义语言,定义对数据库对象(库、表、列、索引)的操作。
CREATE、DROP、ALTER、RENAME、 TRUNCATE等
DML(Data Manipulation Language): 数据操作语言,定义对数据库记录的操作。
INSERT、DELETE、UPDATE
DCL(Data Control Language): 数据控制语言,定义对数据库、表、字段、用户的访问权限和安全级别。
GRANT、REVOKE等
Transaction Control:事务控制
COMMIT、ROLLBACK、SAVEPOINT等
(1)常用的数据类型 数值型(int、number)、字符型(char(固定长度)、varchar(可变长度)
varchar2(可变长度))、日期型date
(2)DDL(数据定义语言:表结构) 定义对数据库对象(库、表、列、索引)的操作
建表: create table tb_person( id int, name varchar2(15), sex char(3), age int ); 查询表: select * from tb_person; 删除表: drop table tb_person; 修改表; 增加列: alter table tb_person add(phone varchar2(18),address varchar2(30)); 改变列长度: alter table tb_person modify(phone varchar2(20)); 删除列: alter table tb_person drop(phone,address); 修改列名: alter table tb_person rename column phone to iphone7s; 修改表名: rename tb_person to tb_people; 截断表 当表结构必须保留, 而表数据不再需要的时候 TRUNCATE TABLE tb_person
(3)DML(数据操作语言) 定义对数据库记录的操作
插入: insert into tb_person(id,name,sex,age) values(1,'tom','男',20); 删除数据: delete from tb_personn;或者DDL中的截断: truncate table tb_person; 修改数据: update tb_person set age=21; update tb_person set age=21,id=2 where sex='男' and name='tom';
sql语句删除数据有几种方式?分别是什么?
答: 1. delete -- DML语句,delete语句最小单位是行,可以通过where选择删除,删除数据可以回滚, 保留表结构。最慢 2. truncate -- DDL语句,TRUNCATE是截断表的所有数据,删除数据不可以回滚,保留表结构。较快 3. drop -- DDL语句,删除表结构和表所有数据,并且不能回滚,慎用。最快
(4)事务控制
transaction control:事务控制 commit(提交)、rollbck(回滚)
(4)约束
check 检查约束 -- not null 非空约束(特殊的检查约束) unique 唯一约束 primary key 主键约束(非空且唯一) foreign key 外键约束(解决数据冗余问题) 列级约束: create table tb_person( id int primary key, name varchar2(18) not null, sex char(3) check(sex='男' or sex='女'), age int check(age>=18 and age<60), phone varchar2(20) unique, address varchar2(30), clazz_id int references tb_clazz(id) on delete cascade ); create table tb_clazz( id int primary key, code varchar2(15), name varchar2(15), bzr varchar2(20) );
外键中,当主表的记录被子表参照时,主表记录不允许被删除。 解决方案: 先删除字表数据,在删除父表数据。 先将字表关联数据设为null,在删除父表数据 先将子表数据修改,在删除父表数据。 on delete cascade 当删除父表数据时,级联删除子表数据。 on set null 当删除父表数据时,将关联的数据设置为null
表级约束: create table tb_student( id int, name varchar2(18), sex char(3), age int, phone varchar2(18), email varchar2(18), address varchar2(18), clazz_id int, constraints tb_students_pk primary key(id), check(name is not null), check(sex='男' or sex='女'), check(age>18 and age<50), unique(phone), unique(email), constraints tb_students_fk foregin key(clazz_id) reference tb_clazz(id) ); 自定义约束名:(约束 约束名 约束类型) constraints tb_students_pk 数据库建议建表: 外键以表级约束,其他用列级约束。 约束的维护(添加、删除): alter table tb_student add primary key(id); 删除约束根据约束名删除: alter table tb_student drop constraints tb_students_pk; 复合约束(只能在表级中定义): primary key(year,month) (5)nvl函数 nvl(第一个参数,第二个参数), 如果第一个参数为null,则取第二个参数 定义字段的别名as select empno as eID,ename,sal,sal*12 as yearsal from scott.emp; distinct关键字去除重复数据: select distinct deptno from scott.emp; 比较运算符: select *from scott.emp where sal>=800 and sal<=1600; select * from scott.emp where sal between 800 and 1600; select * from scott.emp where deptno=20 or deptno=30; select * from scott.emp where deptno in(20,30); 模糊查询like: %匹配所有 _匹配一个字符 select * from scott.emp where ename like 's%'; select * from scott.emp where ename like '%s'; select * from scott.emp where ename like '%s%'; select * from scott.emp where ename like '_s%'; 优先级规则:先and后or 对结果排序 order by asc(升序 默认) desc(降序) select * from scott.emp order by sal asc; select * from scott.emp order by sal desc; (6)多表连接 内连接(等值连接) 两个表(或连接)中某一数据项相等的连接称为内连接。 SELECT FROM dept d INNER JOIN emp e ON d.deptno = e.deptno'; 外连接(非等值连接) 用于查询一张表在另一张表中没有关联数据的信息 外连接分为三种: 左外连接(LEFT OUTER JOIN) 右外连接(RIGHT OUTER JOIN) 全外连接(FULL OUTER JOIN) -- 左外连接:+号在右边,左边的表的所有数据都要显示,如果右边表没有对应的数据,则补Null -- 右外连接:+号在左边,右边的表的所有数据都要显示,如果左边表没有对应的数据,则补Null -- 全外连接:两张表的所有数据都要全部显示 备份一张表(只备份数据,不备份约束): create table tb_emp as select * from scott.emp; select e.empno,e.ename,e.mgr,t.ename from scott.emp e,tb_emp t where e.mgr=t.empno and e.empno='7369'; 自连接: select e.empno,e.ename,e.mgr,t.ename from scott.emp e,scott.emp t where e.mgr=t.empno and e.empno='7369'; (7)组函数:(组函数都会忽略NULL值) 对一组值进行运算,并返回单个值,也叫聚合函数。 count(*|列名) 统计行数: select count(*) from scott.emp; select count(comm) from scott.emp; sum(数值类型列名) 求和: select sum(sal) from scott.emp; avg(数值类型列名) 平均值: select avg(sal) from scott.emp; max(列名) 最大值: select max(sal) from scott.emp; min(列名) 最小值: select min(sal) from scott.emp; (8)分组 group by 把该列具有相同值得多条记录当成一条记录处理,最后只输出一条记录。 分组函数忽略空值。结果集隐式按升序排列, 如果需要改变排序方式可以使用order by子句。 按部门分组: select deptno from scott.emp group by deptnot; 按工作岗位分组: select job from scott.emp group by job; group by子句的真正作用在于与各种组函数配合使用: select deptno,count(*),sum(sal),avg(sal),max(sal),min(sal) from scott.emp group by deptno;
面试题:数据条件筛选时有几种方式?有什么区别?
限定组的结果:having子句 having子句用来对分组后的结果进行条件过滤。 having和where的区别; where和having都是用于条件过滤。 where在分组前进行条件过滤,having子句是在分组后进行条件过滤, where子句中不能使用聚合函数,having子句可以使用聚合函数。 select deptno,count(*),sum(sal),avg(sal),max(sal),min(sal) from scott.emp group by deptno having sum(sal)>9000 order by sum(sal) desc;
(9)子查询 子查询就是把多条语句写成一条,子查询执行的时候先执行子查询 再执行主查询: select * from scott.emp where sal>( select sal from scott.emp where ename='allen' ); 子查询需要注意的问题:单行子查询返回多行 多行比较运算符: In:与列表中的一个值相等(包含) any:与子查询返回的每一个值比较 all:与子查询返回的所有值比较 in: select * from scott.emp where sal in( select min(sal) from scott.emp group by deptno ); any与子查询返回的每一个值比较 >any 大于最小的 <any 小于最大的 select * from scott.emp where sal>any( select min(sal) from scott.emp group by deptno ); all与子查询返回的所有值进行比较 >all 大于最大的 <all 小于最小的 select * from scott.emp where sal>all( select min(sal) from scott.emp group by deptno ); (10)集合运算:从多个结果集中提取数据 union 把两个结果集的数据合起来,然后干掉重复的 select deptno from scott.dept union select deptno from scott.emp; union all:把两个结果集的数据合起来 select deptno from scott.dept union all select deptno from scott.emp; intersect:把两个结果集的数据合起来,然后干掉重复的, 再找两个查询中都出现的记录 select deptno from scott.dept intersert slect deptno from scott.emp; minus:判断数据存在第一查询的结果集,而·不存在第二查询的数据集 select deptno from scott.dept minus slect deptno from scott.emp; 面试题: select r.id,r.R1,nvl(b.R2,'null') from r,b where r.id=b.id(+) union selet b.id,nvl(r.R1,'null'),b.R2 from r,b where r.id(+)=b.id; 标准外连接: select r.id,r.R1,b.R2 from r left outer join b on r.id=b.id union select b.id,r.R1,b.R2 from r right outer join b on b.id=r.id; 全外连接: select nvl(r.id,b.id) as id,r.R1,b.R2 from r full outer join b on r.id=b.id; (11)rownum 伪列 ‘结果集’中产生的序列 select rownum,deptno,dname,loc from scott.dept; 只有存在rownum=1的记录,才可能存在rownum=2的记录 利用rownum分页: select * from( select rownum as tempid,empno,ename from scott.emp )t1 where t1.tempid between 6 and 10; rowid:一般来说每一行数据对应一个rowid,而且固定且唯一。 在这一行存入数据库时就确定了。可以理解为java对象中的内存地址。 可以利用rowid来查询记录,而且通过rowid查询速度最快的查询方法。 rowid只有在表发生移动(比如表空间变化,数据导入/导出后)才会发生变化。 面试题: 删除重复数据(删除所有重复数据 /删除重复数据但保留一条(保留最大rowid或者最小)): 删除所有数据; delete from tb_test where name in( select name from tb_test group by name,age having count(*)>1 ); 删除数据,保留一条数据(第一种方法); create table tb_tmp as select distinct name,age from tb_test; truncate table tb_test; insert into tb_test(name,age) select name,age from tb_tmp; 删除数据,保留一条数据(方法二): delete from tb_test where rowid not in( select max(rowid) from tb_test group by name,age ); (12)常用函数 dual是oracle提供的一个虚表 select length('hello') from dual; 常用函数: lower把大写转成小写 upper把小写转大写: select upper('helloworld') from dual; select lower('HELLOWORLD') from dual; select * from scott.emp where ename='smith'; select * from scott.emp where lower(ename)='smith'; initcap使串中的所有单词首字母变成大写: select initcap('sql course') from dual; concat 连接两个字符串: select concat('Hello','World') from dual; substr 取字符串,从start开始,取count个: select substr('HelloWorld',1,5) from dual; 从4开始取到末尾: select substr('Helloworld',4) from dual; length 返回字符串的长度: select length('HelloWorld') from dual; instr 在一个字符串中搜索指定的字符,返回发现指定的字符的位置,从1开始: select instr('HelloWorld','l') from dual; trim 删除首尾的空字符串 select trim(' HelloWorld ') from dual; replace 替换: select replace('HelloWorld','ll','ff') from dual; round 四舍五入; select round(45.926,2) from dual; trunc 截断: select trunc(45.926,2) from dual; mod 取模: select mod(1600,300) from dual; decode函数: select ename,job,sal 基本工资,decode(job, 'salesman', sal*0.9, 'manager',sal*0.85, 'clerk',sal+100, 'pre sident',sal )as 实发工资 from scott.emp;
面试题: select name, sum(decode(t.course,'JDBC',t.grade,0)) JDBC, sum(decode(t.course,'Hibernate',t.grade,0)) Hibernate, sum(decode(t.course,'Spring',t.grade,0)) Spring from tb_course t group by t.name; 11.数据库对象 (1)数据库对象包括: 表、序列、同义字、数据库链接、视图等。 对象的特点:可以给其他对象使用,名字不能重复。 (2)序列:自动生成的唯一序列号,常用在主键自动生成。 创建序列: create seqeuence s_tb_student; 删除序列: drop seqeuence s_tb_student; 序列的两个属性currval,nextval: select s_tb_student.currval from dual; select s_tb_student.nextval from dual; 使用序列: insert into tb_student(id)values(s_tb_student.nextval); 序列不会随着rollback回滚,下面这些情况时sequence值可能产生间隙: 回滚发生; 序列用于多个表(不建议),建议一张表使用一个序列; 系统宕机。 (3)索引 面试:如何优化你的数据库查询? 1.数据库的查询方式? --全表扫描 select * from tb_student 慢 --利用索引扫描 快 --共享语句 最快(oracle有个回滚段,临时表空间) 索引 index 作用:在数据库中用来加速对表的查询 原理:通过使用快速路径访问方法快速定位数据,减少了磁盘的I/O 特点:与表独立存放,但不能独立存在,必须属于某个表 由数据库自动维护,表被删除时,该表上的索引自动被删除。 索引的创建: 自动:当在表上定义一个primary key或者unique约束1条件时,数据库会 自动创建一个对应的索引。 手动:用户可以创建索引以加速查询。 create index i_tb_student_name on tb_student(name); 查询的时候使用索引: select * from tb_student where name='Alice'; 删除索引: drop index i_tb_student_name; 当创建索引的时候,oracle会默认创建一个和当前表相关的索引页, 而索引页中保存了索引字段和真实的磁盘地址,当用户发送sql语句带了索引时, oracle会到索引页中查询索引字段,直接定位磁盘IO,提取数据。 所以索引数据快于全表扫描。 索引的维护 1.建立索引后,查询的时候需要在where条件中带索引字段才可以使用索引。 2.在经常查询的字段上面建立索引。不要在所有字段上建立索引。 3.因为索引是用来加速查询速度的,如果一张表经常做insert,delete, update ,而很少做select,不建议建立索引 如果一张表字段很少,不建议建立索引。 4.索引是由oracle自动维护的。索引使用久了会产生索引碎片 (磁盘碎片),影响查询效果,所以使用久了需要手动进行维护(删除在重建) sql语句的优化: 多使用共享语句,尽量使你的sql语句能够使用索引。 怎样使sql语句能够使用到索引呢? 当sql语句中包含not in,<>,is null,is not null,like'%%'的时候 不会使用索引。 in:可以使用索引。 优化方案: a<>0 改为a>0 or a<0 a is not null改为 a>0 或a>'' is null 用一个缺省值代替空值 like'%%'不能使用索引 like'001%'可以使用索引 (4)同义字 :通过创建同义字(对象的另外的一个名字)简化访问对象的操作。 create synonym tb_emp for scott.emp; select * from tb_emp; 删除:drop synonym tb_emp; 数据库链接--database link (5)视图 创建用户时通过dba角色赋予权限,而不是手动赋予权限,则会创建 视图会报:没有权限 解决方案:使用system用户登录,在给j1703赋予权限 grant create any table to j1703 with admin option; grant create any view to j1703 with admin option; grant select any table to j1703 with admin option; 所有数据字典都是视图: select * from User_Tables; 视图可以使复杂的查询变得简单 创建简单视图(一个表): create view v_deptinfo as select deptno as 部门,count(*) 人数,sum(sal) 总工资,avg(sal) 平均工资,max(sal) 最高工资,min(sal) 最低工资 from scott.emp group by deptno; 使用视图:select *from v_deptinfo; 复杂视图(多个表) 删除视图:drop view v_deptinfo; (6)数据建模(试规范题) 软件开发过程: 1. 需求调研,与客户进行沟通 2. 需求分析,将现实工作中的动作模拟到计算机 数据建模 3. 开发 4. 测试 5. 上线部署 从关系数据库的表中删除冗余信息的过程称为规范化, 是得到高效的关系型数据库表的逻辑结构最好和最容易的方法。 规范化数据时:应执行以下操作: 将数据库的结构精简为最简单的形式 从表中删除冗余值 标识所有依赖与其他数据的数据 获得数据化的三种方法:三范式 步骤1: 第一范式:必须要有主键,并且每个属性值都是不可分的最小数据单元 ,则称为是第一范式。 第二范式:所有非主关键字都完全依赖于主关键字(通常用于联合主键) 第三范式:非主关键字不能依赖于其他非主关键字(通常用于一个主键) 数据建模 1.根据三个范式 2.分析实体之间的关系 1对1:一个人只有一个身份证 唯一外键关联或者主键关联 一对多:一个班级可以有多个学生 一个学生只属于一个班级 关联:一对多使用主外键关联,通常在多方建立外键 多对多:一个学生可以选择多门课程 一门课程可以被多个学生选修 关联:多对多通常使用中间表(再多建一张表存储)关联数据 通常中间表会有两张表的id作为联合主键,并且 作为外键指向关联表 订单和用户之间的关系是多对一: create table tb_order( id int primary key, code varchar2(50), user_id int, foregin key(user_id) references tb_user(id) ); 订单和书籍是多对多关系: create table tb_item( order_id int, book_id int, count int, primary key(order_id,book_id), foregin key(order_id) references tb_order(id), foregin key(book_id) references tb_book(id) ); select * from tb_user u,tb_book b,tb_order o,tb_item i where u.id=o.user_id and o.id=i.order_id, and b.id=i.book_id
and u.name='jack'; 12.plsql(数据库编程语言) PL/SQL=传统SQL+结构化流程控制 好处:1.有结构化的流程控制,可以完成复杂的操作 2.性能高于sql sql:页面输入信息 -- servlet接收数据,调用持久层生成sql - sql语句发送到数据库(DBMS),数据库编译 -- 执行sql。 plsql:页面输入信息 -- servlet接收数据,调用持久层传入参数(调用存储过程) - 执行sql。 3.可以对程序中的异常进行处理 PL/SQL程序由三个块组成。 (1) 声明部分: 在此声明PL/SQL用到的变量,类型及游标,以及局部的存储过程和函数 (2)执行部分: 过程及SQL 语句 , 即程序的主要部分 (3) 异常处理部分: 错误处理 语法:declare --声明部分,用来定义变量等 begin--可以理解成java的花括号,编写代码的地方 exception--处理程序抛出异常 end; 第一个plsql,在控制台输出HelloWorld BEGIN --包名.过程('参数') dbms_output.put_line('HelloWorld!'); END; 定义变量,变量初始值都是null 常量constant declare v_id int; v_name varchar2(18):='jack'; v_clazz constant varchar2(5):='j1703'; begin v_id:=100; dbms_output.put_line('v_id='||v_id); dbms_output.put_line('v_name='||v_name); end; 复合变量(数据类型) record 1.包含多个内部组件,用于存放多个值 2.需要先定义类型,然后用该类型可重复定义多个变量 注意:复合变量属于数据类型,定义时前面要加TYPE TYPE是不能直接使用的,使用前需要定义变量引用 declare type r_tb_clazz is record ( id int, code varchar2(18) ); v_clazz r_tb_clazz; begin v_clazz.id:=1; v_clazz.code:='j1703'; end; plsql分两大类:匿名块,带名块 begin insert into tb_clazz(id,code) values(3,'j1703'); commit; end; select 1.缺少into子句,plsql目的是操作数据,需要定义变量 2.实际返回行数超过请求行数(只能查询一条数据,如果需要查询 多条数据,使用游标) 3.未找到数据 declare v_id int; v_code varchar2(18); begin select id,code into v_id,v_code from tb_clazz where id=1; dbms_output.put_link(v_id || '' || v_code); end; %type属性: 定义某个变量的数据类型与已经存在的变量数据类型, 某个列的数据类型相同。 declare v_id tb_clazzz.id%type; v_code tb_clazz.code%type; begin select id,code into v_id,v_code from tb_clazz where id=1; dems_output.put_line(v_id || '' || v_code); end; %rowtype属性: 用于定义不确定的类型的变量 当数据库中表字段的个数和数据类型会在运行中改变, 程序中的变量也会自动随之改变。 rt_tb_clazz tb_clazz%rowtype rt_tb_clazz 变量和tb_clazz表结构一致,相当于表结构的副本。 declare v_tb_clazz tb_clazz%rowtype; begin select id,code into v_tb_clazz.id,v_tb_clazz.code from tb_clazz where id=1; dbms_output.put_line(v_tb_clazz.id || '' || v_tb_clazz.code); end; declare v_tb_clazz tb_clazz%rowtype; begin select * into v_tb_clazz from tb_clalzz where id=1; dbms output.put_line(v_tb_clazz.id || '' || v_tb_clazz.code); end; 使用recode declare type r_tb_clazzz is recode ( id tb_clazz.id%type, code tb_clazz.code%type; ); v_tb_clazz r_tb_clazz; begin select * into v_tb_clazz from tb_clazz where id=1; dbms_output.putline(v_tb_clazz.id || '' || v_tb_clazz.code); end; 循环和判断 if分支 declare v_i number:=10; begin if(v_i=10)then dbms_output.putline('进入if块'); end if; end; if else分支 if ()then...else ...end if if elsif else分支 if() then...elsif()then ...else ...end if plsql的循环有三种:loop for while loop循环: loop v_i:=v_i+1;--自增 exit when v_i=5; end loop; 第二种结束方法: loop v_i:=v_i+1; if(v_i=5) then exit; end if; end loop; for循环 for v_i in 1..5 loop end loop; --重点 项目当中最常用的异常处理 --实际开发中异常的处理 重点 create table tb_error( id int primary key, errorObj varchar2(18),--抛出异常的对象 sqlcode varchar(50),--异常编码 sqlerrm varchar(200),--异常信息 currdate date --发生时间 ); create sequence s_tb_error; --重点掌握 --orale当抛出异常时会将异常编码存储到sqlcode函数,异常信息 -- 存储到sqlerrm declare v_id tb_clazz.id%type; v_code tb_clazz.code%type; begin select id,code into v_id,v_code from tb_clazz; dbms_output.put_line(v_id || '' || v_code ); exception when others then v_sqlcode :=sqlcode; v_sqlerrm:=sqlerrm; insert into tb_error(id,errorobj,sqlcode,sqlerrm,currdate) values(s_tb_error.nextval,'default',v_sqlcode,v_sqlerrm,sysdate); commit; end; --游标:用来提取多行数据 --oracle打开一个工作区(缓存)来保存多行查询的结果集, -- 游标就是给这个工作区命的名称,并能用于处理由多行查询而返回的 记录行 --隐式游标:默认的DML语句和select语句都有隐式游标 --显示游标:开发中给工作区命名,并且可以进行操作 --%isopen boolean 游标打开,则返回true --%nofound boolean 如果最近抓取没有获得记录,返回true --%found boolean 如果最近抓取获得记录,返回true --%rowcount number 返回到目前为止获取的记录数 --使用游标的步骤 --1.定义游标 cursor c_tb_clazz is select * from tb_clazz; --2.打开游标 open c_tb_clazz; --3.fetch游标 fetch c_tb_clazz into r_tb_clazz; -- 游标有个指针,默认指向第一行之上,fetch将指针向下移动,指向第n 行数据,如果有数据,notfound 返回false,found返回true 如果到末尾,抓取不到数据,一直显示最后一条数据 --4.关闭游标 close c_tb_clazz; --第一个例子 ;使用游标提取tb_clazz的所有数据 declare cursor c_tb_clazz is select * from tb_clazz; r_tb_clazz tb_clazz%rowtype; begin open c_tb_clazz; loop fetch c_tb_clazz into r_tb_clazz; dbms_output.put_line(r_tb_clazz.id || '' || r_ tb_clazz.code); end loop; close c_tb_clazz; end; --带参数的游标 第二个例子:使用游标提取tb_clazz的所有数据,同时提取每个班级的学生数据 --注意:传递是形参,形参是不用长度的 declare --班级游标 cursor c_tb_clazz is select * from tb_clazz; r_tb_clazz tb_clazz%rowtype; --学生游标 cursor c_tb_student(v_clazz_id tb_clazz.id%type) is select * from tb_student where clazz_id=v_clazz_id; r_tb_student tb_student%rowtype; begin open c_tb_clazz; loop fetch c_tb_clazz into r_tb_clazz; exit when c_tb_clazz%notfound; dbms_output.put_.line('班级' || r_tb_clazz.id || '' || r_tb_clazz.code); --当前班级的学生数据 --打开学生游标,参数是班级id open c_tb_student(r_tb_clazz.id); loop fetch c_tb_student into r_tb_student; exit when c_tb_student%nofound; dbms_output.putline(r_tb_student.id || '' || r_tb_student.name); end loop; close c_tb_student; end loop; close c_tb_clazz; end; --游标for循环 1.不需要显式打开游标 2.for循环自动隐式地定义recode变量 3.不需要使用fetch抓取数据 4.循环结束后,不需要使用close来关闭游标 declare cursor c_dept is select * from scott.dept; begin for re_dept in c_dept loop dbms_output.put_line(); end loop; end; --带名块(declare begin end; 就是匿名块) 是数据库中命名的PL/SQL块,作为数据库对象保存在数据库里。 主要四类: 存储过程:执行特定操作,无返回值 函数:进行复杂计算,有返回值 包:逻辑上相关的过程和函数组织在一起 触发器:事件触发,执行相应操作 过程和函数统称为PL/SQL子程序,过程和函数的唯一区别是函数总向调用者返回数据, 而过程则不返回数据。 --存储过程 语法:(直接创建) procedure name [(parameters)形参] is 局部变量声明 begin 语句; [exception] end; --建立存储过程,可以被多个程序调用(java或者c++),可以向 --存储过程传递参数,也可以让存储过程传出参数 --参数: --in:默认,值被传递给子程序。子程序不能改变参数值。 --out:值被返回调用环境(java或者c++),子程序会改变参数值。 --调用存储过程 begin insertclazz();--存储过程名 end; --调用存储过程通过带参数传给存储过程 --函数与过程语法差异: (1)在函数的声明中,必须包含一个带有数据类型的return字句, 返回值表示的类型,在函数体中必须有一个有效的return语句。 (2)只能使用in模式参数传入参数值 --语法: function name [(parameters)形参] return datatype is 局部变量声明 begin &