[size=medium]
703
----------------------------------包package-------------------------------------
--Package概念:按照业务逻辑、把相关的Func , Procedure 组织到一起,形成一个函数或者过程集合
--package组成:包说明(package specification)、包体(package body)
--package好处:
1、模块化:一般把有相关性的函数和过程放到一个Package中;
2、易设计:可以把包说明和包体分别编写和编译,先编写和编译包说明部分,在编写和说明包体部分;这 有利于分工合作;
3、信息隐藏:包体中函数可以部分出现在包说明中,只有出现在包说明中的函数和过程才是该Package的 公有函数和过程,可以被其他包中的函数调用,否则对其他包中的函数是不可见的,未在包说 明部分出现的函数和过程相当于私有的。
4、加载性能提高:当Package中有一个函数或过程被调用时,整个Packege就被加载到内存中,这样当该
Package中其他函数被调用时,就直接从内存读取了,可以减少磁盘IO,从而提高性能。这 个特性也提醒我们不要去搞巨无霸的Package, 把你用到的任何函数都写到一个Package中 这会导致严重的内存浪费。
5、重载:一个package 中可以定义同名、不同参数的函数或过程。
--package向前声明特性
在Package body中,一个函数中调用另一个函数(也在该Package中),则另一个函数必须在前面先定义; 如果你非要调用在程序代码中后定义的函数,可把这个函数设置成公有函数,在包说明部分说明;
--初始化过程代码
session加载时被执行一次
一般用于一些复杂变量的初始化(比如某个公有变量的初始化值是需要通过一段负责的SQL来获取的)
不需要则NULL
--package中的变量的持久化状态
package中的各个变量,不同的session不会相互影响。
---------------------------------------------------------------------------------
------------------------------------内置SQL工具包-------------------------------
---DBMS_SQL 动态SQL
1、DBMS_SQL
CREATE OR REPLACE PROCEDURE delete_all_rows(p_tab_name IN VARCHAR2,
p_rows_del OUT NUMBER) IS
cursor_name INTEGER;
BEGIN
cursor_name := DBMS_SQL.OPEN_CURSOR;
DBMS_SQL.PARSE(cursor_name,
'DELETE FROM ' || p_tab_name,
DBMS_SQL.NATIVE);
p_rows_del := DBMS_SQL.EXECUTE(cursor_name);
DBMS_SQL.CLOSE_CURSOR(cursor_name);
END;
2、EXECUTE IMMEDIATE
CREATE PROCEDURE del_rows(p_table_name IN VARCHAR2, p_rows_deld OUT NUMBER) IS
BEGIN
EXECUTE IMMEDIATE 'delete from ' || p_table_name;
p_rows_deld := SQL%ROWCOUNT;
END;
--DBMS_DDL 程序中执行DDL
1、在程序中执行编译命令
DBMS_DDL.ALTER_COMPILE('PROCEDURE','A_USER','QUERY_EMP')
2、在程序中执行数据收集命令
DBMS_DDL.ANALYZE_OBJECT('TABLE','A_USER','JOBS','COMPUTE')
--DBMS_JOB 定义job定期执行某个程序
job的定义、提交、更改、停止、移除
DECLARE
jobno NUMBER;
BEGIN
DBMS_JOB.SUBMIT(job => jobno,
what => 'OVER_PACK.ADD_DEPT(''EDUCATION'',2710);',
next_date => TRUNC(SYSDATE + 1),
interval => 'TRUNC(SYSDATE + 1)');
dbms_output.put_line(‘job_no = ‘ || jobno) COMMIT;
END;
查找自己提交的job号
SELECT job, log_user, next_date, next_sec, broken, what FROM DBA_JOBS;
--UTL_FILE 读写外部文件
。。。。。
-----------------------------------------------------------------------------------
--------------------------------------plsql中大对象的操作---------------------------
--oracle数据库中的lob类型:
1、CLOB :字符大对象,存储在数据库内部;
2、NCLOB:多字节字符大对象,存储在数据库内部;
3、BLOB:二进制大对象,存储在数据库内部;
4、BFILE:二进制文件,存储在数据库外部;
--内部LOB的一般操作步骤
1、在表中添加LOB类型的列
2、在程序中声明和初始化LOB的Locator
3、使用SELECT FOR UPDATE 锁定目标行,准备更新行上的LOB列(LOB的Locator)
4、生成LOB对象,可使用DBMS_LOB 这样的PLSQL包,也可以使用OCI,JDBC等;
5、Commit 提交更改;
--外部LOB BFILE的操作
Bfile是数据库外部文件,在数据库表上这种类型的字段实际只是存储一个Locator
Bfile的使用限制:Bfile是数据库外部文件,是只读的,所以不参与事务操作;用户必须先创建文件并放到
特定的目录下,给予Oracle进程以目录和文件的读取权限;
--oracle directory
控制Bfile存储的安全性
使用Bfile的一般步骤:
1、在操作系统上创建目录,并给Oracle数据库进程赋予阅读权限,把外部文件放入这个目录
2、在Oracle数据库中表添加Bfile类型字段
3、在Oracle 数据库中创建Directory 对象
CREATE DIRECTORY dir_name AS os_path;
4、授权读权限给特定的数据库用户
GRANT READ ON DIRECTORY dir_name TO user|role|PUBLIC;
5、往表中插入数据时使用BFILENAME 函数,它可以关联外部文件和表上的Bfile列
6、在程序中声明和初始化LOB的Locator
7、Select 指定行上Bfile 列到Locator
8、使用DBMS_LOB 或者通过OCI 读取Bfile (使用Locator作为文件的一个引用)
--DBMS_LOB
1、更改LOB的值: APPEND, COPY, ERASE, TRIM, WRITE, LOADFROMFILE
2、读取、检查LOB的值: GETLENGTH, INSTR, READ, SUBSTR
3、Bfile专用:FILECLOSE, FILECLOSEALL, FILEEXISTS, FILEGETNAME, FILEISOPEN, FILEOPEN
---------------------------------------------------------------------------------------
-------------------------------------oracle触发器trigger-------------------------------
--关键因素
时机:Before 或者After 或Instead of
事件:Insert 或Update 或Delete
对象:表名(或视图名)
类型:Row 或者Statement级;
条件:满足特定Where条件才执行;
内容:通常是一段PLSQL块代码;
-Instead of : 用Trigger的内容替换事件本身的动作
-Row级:SQL语句影响到的每一行都会引发Trigger
-Statement级:一句SQL语句引发一次,不管它影响多少行(甚至0行)
--创建statement级别Trigger语法
CREATE [OR REPLACE] TRIGGER trigger_name
timing
event1 [OR event2 OR event3]
ON table_name
trigger_body
CREATE OR REPLACE TRIGGER secure_emp
BEFORE INSERT OR UPDATE OR DELETE ON employees
BEGIN
IF (TO_CHAR(SYSDATE, 'DY') IN ('SAT', 'SUN')) OR
(TO_CHAR(SYSDATE, 'HH24') NOT BETWEEN '08' AND '18') THEN
IF DELETING THEN
RAISE_APPLICATION_ERROR(-20502,
'You may delete from EMPLOYEES
table only during business hours.');
ELSIF INSERTING THEN
RAISE_APPLICATION_ERROR(-20500,
'You may insert into
EMPLOYEES table only during business hours.');
ELSIF UPDATING('SALARY') THEN
RAISE_APPLICATION_ERROR(-20503,
'You may update
SALARY only during business hours.');
ELSE
RAISE_APPLICATION_ERROR(-20504,
'You may update
EMPLOYEES table only during normal hours.');
END IF;
END IF;
END;
--创建row级别trigger
CREATE [OR REPLACE] TRIGGER trigger_name
timing
event1 [OR event2 OR event3]
ON table_name
[REFERENCING OLD AS old | NEW AS new]
FOR EACH ROW
[WHEN (condition)]
trigger_body
CREATE OR REPLACE TRIGGER audit_emp_values
AFTER DELETE OR INSERT OR UPDATE ON employees
FOR EACH ROW
BEGIN
INSERT INTO audit_emp_table
(user_name,
timestamp,
id,
old_last_name,
new_last_name,
old_title,
new_title,
old_salary,
new_salary)
VALUES
(USER,
SYSDATE,
:OLD.employee_id,
:OLD.last_name,
:NEW.last_name,
:OLD.job_id,
:NEW.job_id,
:OLD.salary,
:NEW.salary);
END;
--INSTEAD OF Trigger
[/size]