在Oracle8i之前,开发人员只能使用PL/SQL来开发存储过程。而在Oracle8i之中,不仅可以使用原有的PL/SQL开发存储过程,而且也可以使用Java语言来开发存储过程。本篇文章将简单介绍关于这方面的知识,包括以下内容:(灰色显示的部分在此将不作详细介绍)
本文是我参考网上资料,整理出来的一份,文中所有代码语句均真实可用,因只是作测试,所以代码未表现实际业务操作。
七. 使用Java开发触发器
八. 使用Java开发对象方法
九. 使用JDeveloper开发JSP
一. 存储过程简介
存储过程是存储在数据库中的一段存储程序。当创建存储过程时,系统会对其进行编译,并将执行代码存储到数据库中。
1. 设计存储过程的方针
1) 在定义存储过程时,要使用其完成单一、相对集中的任务。
2) 在定义存储过程时,不要定义已经由其它特征所提供功能的过程。例如,不要定义强制数据完整性的过程(使用完整性约束)。
2. 存储过程的优点
1) 安全性
当创建了存储过程之后,可以将执行该过程的权限授予其它用户,从而使得他可以执行特定的数据库操作,而不能访问其它模式对象(例如表)。例如,你可以将执行过程(更新表)的权限授予其它用户,但不授予它们直接访问该表的权限。
2) 性能
① 存储过程只被发送到数据库一次,相对于SQL语句或PL/SQL块而言,其网络通信量更小。
② 当调用存储过程时,数据库会直接运行该存储过程,无需进行编译。相对于SQL语句或PL/SQL块而言,其执行速度更快。
3) 内存分配
存储过程充分利用了Oracle共享内存的能力。在将存储过程装载到内存中后,多个用户可以同时调用该存储过程,从而降低了应用对Oracle的实际内存需求。
4) 生产力
存储过程提高了开发生产力。通过将公共集合编写为存储过程,避免了冗余代码,从而提高了开发生产力。例如,我们可以编写用于插入、更新、删除DEPT表的过程,此后应用可以直接调用这些过程,而无需重写SQL语句。当管理数据的方法发生变化时,只需要修改过程,而不需要对应用进行任何修改。
二. Java存储过程
在以前的Oracle版本中,开发存储过程是通过PL/SQL来完成的。而在Oracle8i版本中,我们不仅可以使用PL/SQL开发存储过程,而且还可以使用Java语言来开发存储过程。
1. PL/SQL与Java存储过程比较
与PL/SQL相比,使用Java语言开发存储过程有以下优点:
1) Java语言具有更强大的运算能力,提供了更多的运算方法。当要完成进行复杂运算的存储过程时,使用JSP(Java Storage Procedure)将是你最好的选择。
2) PL/SQL只能用于Oracle数据库,而Java语言可以应用于更多的数据库系统(如Sybase、DB2、Informix等等),所以Java存储过程将具有更好的兼容性、可移植性。
2. JSP(Java Storage Procedure)分类
Java存储过程包括过程、函数、触发器以及对象方法四种类型。
3. 调用JSP的四种方法
1) CALL语法
2) DML语句
3) PL/SQL块、子程序、包
4) 由触发器隐含调用
三 Java存储过程的开发步骤
1. 编写Java源代码
当开发Java存储过程时,首先应该编写Java源代码。
注意事项:
① 一般都写成public static方法作为Java存储过程
② 如果从SQL调用Java的实例方法。就必须在Oracle中建立Oracle Objects对象,用Java方法实现对象的方法。这个技术使用CREATE TYPE BODY命令。
2. 装载Java代码及类到Oracle8i数据库中
在编写了Java源代码之后,接下来应该将Java代码及相应的Java类装载到Oracle8i数据库中。
装载Java代码及类到RDBMS有以下两种方法:
1) 使用loadjava工具,通过该工具可以快速装载Java源代码(.java)、Java二进制代码(.class)以及Java打包文件(.jar)。
2) 使用CREATE Java、ALTER Java装载Java代码。
其中,前一种方法相对简单,并且我们推荐你使用这种方法。 如果借助于像PL/SQL Developer和JDeveloper来开发JSP装载过程就更简单,比如在PL/SQL DEveloper的SQL窗口第一行加上 "create or replace and compile java source named Foo as" 点执行工具按钮,PL/SQL Developer就知道是要把Java代码装载到Oracle中
3. 生成调用说明
在装载了Java类之后,接下来应该生成对public static方法的调用说明,最终完成Java存储过程的开发工作。就是声明哪些Java类方法要暴露给数据库调用。
完成上述步骤之后,就完成了Java存储过程的开发工作,然后就可以调用并执行该Java存储过程了。
下面将逐一分绍用Java开发几种存储过程的步骤,实例中的代码及操作均可在 Oracle 8.1.6 中成功执行。
四:使用Java开发过程
过程用于执行某种操作。需要注意的是,过程所对应的Java方法返回值必须为空(void)。
下面讲述完成上述任务的方法及过程: (前三个步骤在开发函数、包、触发器、对象方法基本一致,所以在以后的介绍中将不再重复)
1. 编写Java源代码
程序清单如下(HelloJSP.java):(这段代码在开发其他几种过程的时候也要用到)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 |
import java.io.*; import java.net.*; import java.sql.*; import oracle.jdbc.driver.OracleDriver; public class HelloJSP { // 有返回值,要说明为数据库的函数,有参数 public static String javaFunction(String userName) { // 取系统、数据记录信息等返回 return "Hello " + userName; } // 有返回值,要说明为数据库的函数,无参数 public static String javaFunction1() { try { //通过URL连接读取http://java.sun.com:80上的内容 // 注意,执行前需要指定用户对Socket java.sun.com:80的访问权限,解析与连接 // 在sqlplus中以system用户调用dbms_java包的grant_permission过程,命令如下: // call dbms_java.grant_permission('SCOTT','java.net.SocketPermission', // 'java.sun.com:80', 'resolve,connect, accept'); URL url = new URL("http://java.sun.com"); InputStream is = url.openStream(); BufferedReader br = new BufferedReader(new InputStreamReader(is)); //返回读取到的第一行字符串 return br.readLine(); } catch (Exception e) { return e.getMessage(); } } // 无返回值,要说明为数据库的过程 // 执行成功后将会在c:/看到一个文件a,内容为你所输入的参数字符吕串 public static void javaProcedure(String userName) { // todo something,如维护数据记录,读取系统信息等 try { // 取得连接就可以做你要的数据库操作,用默认连接 Connection conn = new OracleDriver().defaultConnection(); conn.close(); } catch (SQLException e) { } try { // 注意,执行前需要指定用户对文件c:/a至少有写的权限 // 在sqlplus中以system用户调用dbms_java包的grant_permission过程,命令如下: // call dbms_java.grant_permission('SCOTT','java.io.FilePermission','c:/a','read,write'); OutputStream os = new FileOutputStream("c:/a"); os.write(userName.getBytes()); os.close(); } catch (IOException e) { } } public static void javaTrigger(int deptNo, String oldName, String newName) { // to do something // 我们可以直接调用上面的方法,也往产生一个文件c:/a,看看效果 try { javaProcedure("Dept No: "+deptNo+" Old name: "+oldName+" New name: "+newName); } catch (Exception e) { } } } |
2. 装载Java代码及类到Oracle8i数据库中
在编写了Java源代码之后,就可以将Java对象装载到Oracle8i数据库中了。下面是完成这项任务的方法:
你的Java类中用到了哪些包,就应设置好相应的classpath环境变量, 例如:用以下命令
set classpath=%classpath%;%ORA_HOME%/jdbc/lib/classes12.zip;%ORA_HOME%/sqlj/lib/runtime.zip;%ORA_HOME%/sqlj/lib/translator.zip
实际操作loadjava时并没有你上面那样设置classpath也行,当前的classpath是CLASSPATH=.;E:\Oracle\Ora81\orb\classes\yoj.jar;E:\Oracle\Ora81\orb\classes\share.zip,其中这两个包也没有包含上面代码引入的类
装载命令用:
loadjava -user scott/tiger@orcl -o -v -f -r d:/jsp/HelloJSP.java
参数说明:
-o 使用OCI8 JDBC接口
-v 显示执行过程
-f 强制装载
-r 编译并解析类
最后一个参数就是源文件路径
3. 发行Java,生成调用说明
在装载了Java类后,就可以发行该Java类,并生成调用其方法的过程说明了(当然首先要登录到数据库了)
1)
1 2 3 4 |
CREATE OR REPLACE PROCEDURE JavaProcedure(userName IN VARCHAR2) AUTHID CURRENT_USER AS LANGUAGE JAVA NAME 'HelloJSP.javaProcedure(java.lang.String)'; / |
4. 调用JSP
在生成了调用Java方法的过程说明之后,我们就可以调用JSP了。例如:
call JavaProcedure('Unmi');
成功执行后将会在C盘根目录下产生一个文件a,内容为"Unmi".注意必须赋于SCOTT对文件c:/a的写权限,后面会有一个单独专题来讲解OracleJVM权限。
五. 使用Java开发函数
函数用于返回特定数据。说明使用Java开发函数的方法。与使用Java开发过程不一样的就在发布调用说明与调用略有不同。前两步与上同
3. 发行Java,生成调用说明(先也要登录到数据库了)(下面分别生有有参和无参的函数说明)
1)
1 2 3 4 5 |
CREATE OR REPLACE FUNCTION JavaFunction(userName IN VARCHAR2) RETURN VARCHAR2 AUTHID CURRENT_USER AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction(java.lang.String) return java.lang.String'; / |
和
1 2 3 4 5 |
CREATE OR REPLACE FUNCTION JavaFunction1 RETURN VARCHAR2 AUTHID CURRENT_USER AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction1() return java.lang.String'; / |
4. 调用JSP
在生成了调用Java方法的函数说明之后,就可以调用这些函数了
对函数的调用方式在最顶层可以有两种,call和sql中
1) call dbms_output.put_line(JavaFunction('Unmi'))
2) select JavaFunction1() from dual
六. 使用Java开发包
Java类用于封装Java方法,与此类似,包用于封装过程和函数等。仍然以上面的代码为例。前两步与上同
3. 发行Java,生成调用说明
在装载了Java类后,就可以发行该Java类,并生成调用其方法的包了。 包名为HelloJSP。
1)
1 2 3 4 5 6 7 |
CREATE OR REPLACE PACKAGE HelloJSP as FUNCTION JavaFunction(userName IN VARCHAR2) return VARCHAR2; FUNCTION JavaFunction1 return VARCHAR2; PROCEDURE JavaProcedure(userName IN VARCHAR2); PROCEDURE JavaTrigger(deptNo IN NUMBER,oldName IN VARCHAR2,newName IN VARCHAR2); END HelloJSP ; / |
2)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
CREATE OR REPLACE PACKAGE BODY HelloJSP as FUNCTION JavaFunction(userName IN VARCHAR2) RETURN VARCHAR2 AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction(java.lang.String) return java.lang.String'; FUNCTION JavaFunction1 RETURN VARCHAR2 AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction1() return java.lang.String'; PROCEDURE JavaProcedure(userName IN VARCHAR2) AS LANGUAGE JAVA NAME 'HelloJSP.javaProcedure(java.lang.String)'; PROCEDURE javaTrigger(deptNo IN NUMBER, oldName IN VARCHAR2, newName IN VARCHAR2) AS LANGUAGE JAVA NAME 'HelloJSP.javaTrigger(int, java.lang.String, java.lang.String)'; end HelloJSP ; / |
创建包的主体时,定义函数或过程体时不能加 "AUTHID CURRENT_USER"
4. 调用JSP
在生成了调用Java方法的包后,就可以调用这些方法所对应的函数和过程了。例如:
1) call dbms_output.put_line(HelloJSP.JavaFunction('Unmi'));
2) select HelloJSP.JavaFunction1() from dual;
3) call HelloJSP.JavaProcedure('Unmi');
4) call HelloJSP.JavaTrigger(10,'Kypfos','Unmi');
七. 使用Java开发触发器
触发器是一段存储程序,当执行特定修改操作时,会触发它,并执行其中的存储程序。下面以表dept有记录变动时执行某个操作为例,说明使用Java开发触发器的方法。准确点讲这里所谓的使用Java开发触发器仍然是属于开发储过程的范畴,Java方法仍然是声明为数据库的存储过程。
3. 发行Java,生成调用说明
在装载了Java类后,就可以发行该Java类,并生成调用其方法的过程说明及触发器了。下面是完成该项任务的方法:
从下面两个步骤可以看出,触发器调用Java程序还必须借助于存储过程。由真正的数据库触发器传参到Java存储过程。
1)
1 2 3 4 |
CREATE OR REPLACE PROCEDURE JavaTrigger(deptNo IN NUMBER, oldName IN VARCHAR2, newName IN VARCHAR2) AUTHID CURRENT_USER AS LANGUAGE JAVA NAME 'HelloJSP.javaTrigger(int, java.lang.String, java.lang.String)'; / |
2)
1 2 3 4 5 |
CREATE OR REPLACE TRIGGER CallJavaTrigger AFTER update OR delete OR insert ON dept FOR EACH ROW call JavaTrigger(:old.deptno,:old.dname,:new.dname) / |
4. 调用JSP
在创建了触发器之后,当修改作者工资时会自动调用其中的存储程序。例如:
update dept set dname=dname||'X' where deptno=20
八.使用Java开发对象方法
对象类型是一种用户自定义的数据结构,它可以将数据类型、函数以及过程封装到该数据结构中。对象方法是指对象类型中的函数和过程,
Java对象类必须实现SQLData接口中的getSQLTypeName()、readSQL(SQLInput stream,String typeName)、writeSQL(SQLOut stream)方法
因为在数据库中创建对象在实际开发中比较少用,所以具体的操作方法这里略去,以后用到时再细究。
九. 使用JDeveloper开发JSP
用JDeveloper开发Java存储过程,因为JDeveloper是一个Java IDE,同Oracle又是一家的,能结合的不好吗。基本过程是:
1) 配置好到Oracle数据库的连接
2) 编写好你要嵌入到Oracle中的Java类
3) 工程中New一个Loadjava and Java Store Procedures的deployment Profile
4) 选取要发布类的方法,JDeveloper会根据你的方法有无参数决定是过程还是函数
5) Deploy to 到你配置的数据库上
在Depolyment - Log 中会显示Depoly时的信息,如生成调用说明的语句。
总结:自己按上面的步骤一步步操作,常常会碰到一些问题,而且我都还不知道应该怎么去确定是什么问题,而用jDeveloper来开发Java存储过程,就非常之简单了。而且我还发现JDeveloper给我们开发Java存储过程带来了无比的方便,而且还是一款很棒的Java IDE,支持Struts、JSF、EJB、WebService、TopLink、Swing/AWT、还能画UML图,运行速度也很快。
在JDeveloper中还能够列出已加载到数据库中的Java Classes,而PL/SQL Developer只能看到已加载的Java Source,所以如果用loadjava加载的是一个class文件,在PL/SQL Developer中就是不可见了
。
参考资料:
1. Oracle8i中使用Java语言来开发存储过程(时代朝阳数据库(原晓通数据库)培训部Oracle 资料库) (原文中用的JDeveloper版本是2.0,现在都10.1.3了)
2. 用PL/SQL和Java开发Oracle8 i应用程序
3. 如何利用java过程执行操作系统命令 (ORACLE技术中国用户讨论组)
4. 保证执行Java的安全 (作者:Kuassi Mensah)
5. Permissions in the JDK (Sun官方文档)
本文链接 https://yanbin.blog/oracle8i-develop-java-procedure/, 来自 隔叶黄莺 Yanbin Blog
[版权声明] 本文采用 署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0) 进行许可。
正好要用 ,好文章.