ITPub博客

首页 > Linux操作系统 > Linux操作系统 > pl/sql java

pl/sql java

原创 Linux操作系统 作者:yuanlinok 时间:2010-01-12 11:12:31 0 删除 编辑

在 Oracle8i之前,开发人员只能使用PL/SQL来开发存储过程。而在Oracle8i之中,不仅可以使用原有的PL/SQL开发存储过程,而且也可 以使用Java语言来开发存储过程。本篇文章将简单介绍关于这方面的知识,包括以下内容:(灰色显示的部分在此将不作详细介绍)

本文是我参考网上资料,整理出来的一份,文中所有代码语句均真实可用,因只是作测试,所以代码未表现实际业务操作。

一. 存储过程简介
二. Java存储过程
三. Java存储过程的开发步骤
四. 使用Java开发过程
五. 使用Java开发函数
六. 使用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):(这段代码在开发其他几种过程的时候也要用到)

01.import java.io.*;
02.import java.net.*;
03.import java.sql.*;
04. 
05.import oracle.jdbc.driver.OracleDriver;
06. 
07.public class HelloJSP {
08. 
09. 
10.    // 有返回值,要说明为数据库的函数,有参数
11.    public static String javaFunction(String userName) {
12.        // 取系统、数据记录信息等返回
13.        return "Hello " + userName;
14.    }
15. 
16. 
17.    //  有返回值,要说明为数据库的函数,无参数
18.    public static String javaFunction1() {
19.         
20.        try {
21.            //通过URL连接读取http://java.sun.com:80上的内容
22.            // 注意,执行前需要指定用户对Socket java.sun.com:80的访问权限,解析与连接
23.            // 在sqlplus中以system用户调用dbms_java包的grant_permission过程,命令如下:
24.            // call dbms_java.grant_permission('SCOTT','java.net.SocketPermission',
25.            // 'java.sun.com:80', 'resolve,connect, accept');
26.             URL url = new URL("http://java.sun.com");
27.            InputStream is = url.openStream();
28.            BufferedReader br = new BufferedReader(new InputStreamReader(is));
29.             
30.            //返回读取到的第一行字符串
31.            return br.readLine();
32.        } catch (Exception e) {
33.            return e.getMessage();
34.        }
35.    }
36. 
37.    // 无返回值,要说明为数据库的过程
38.    // 执行成功后将会在c:/看到一个文件a,内容为你所输入的参数字符吕串
39.    public static void javaProcedure(String userName) {
40.        // todo something,如维护数据记录,读取系统信息等
41.        try {
42. 
43.            // 取得连接就可以做你要的数据库操作,用默认连接
44.            Connection conn = new OracleDriver().defaultConnection();
45.            conn.close();
46.        } catch (SQLException e) {
47.        }
48. 
49.        try {
50.            // 注意,执行前需要指定用户对文件c:/a至少有写的权限
51.            // 在sqlplus中以system用户调用dbms_java包的grant_permission过程,命令如下:
52.            // call dbms_java.grant_permission('SCOTT','java.io.FilePermission','c:/a','read,write');
53.            OutputStream s = new FileOutputStream("c:/a");
54.            os.write(userName.getBytes());
55.            os.close();
56.        } catch (IOException e) {
57.        }
58.    }
59. 
60. 
61.    public static void javaTrigger(int deptNo, String oldName,
62.                                   String newName) {
63.        // to do something
64.        // 我们可以直接调用上面的方法,也往产生一个文件c:/a,看看效果
65.        try {
66.            javaProcedure("Dept No: "+deptNo+" Old name: "+oldName+" New name: "+newName);
67.        }
68.        catch (Exception e) {
69.             
70.        }
71.    }
72.}

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.CREATE OR REPLACE PROCEDURE JavaProcedure(userName IN VARCHAR2)
2.       AUTHID CURRENT_USER
3.       AS LANGUAGE JAVA NAME 'HelloJSP.javaProcedure(java.lang.String)';
4.   /

4. 调用JSP

在生成了调用Java方法的过程说明之后,我们就可以调用JSP了。例如:

call JavaProcedure('Unmi');

成功执行后将会在C盘根目录下产生一个文件a,内容为"Unmi".注意必须赋于SCOTT对文件c:/a的写权限,后面会有一个单独专题来讲解OracleJVM权限。

. 使用Java开发函数

函数用于返回特定数据。说明使用Java开发函数的方法。与使用Java开发过程不一样的就在发布调用说明与调用略有不同。前两步与上同

3. 发行Java,生成调用说明(先也要登录到数据库了)(下面分别生有有参和无参的函数说明)

1)

1.CREATE OR REPLACE FUNCTION JavaFunction(userName IN VARCHAR2)
2.      RETURN VARCHAR2
3.      AUTHID CURRENT_USER
4.      AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction(java.lang.String) return java.lang.String';
5.   /

1.CREATE OR REPLACE FUNCTION JavaFunction1
2.      RETURN VARCHAR2
3.      AUTHID CURRENT_USER
4.      AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction1() return java.lang.String';
5.   /

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.CREATE OR REPLACE PACKAGE HelloJSP as
2.        FUNCTION  JavaFunction(userName IN VARCHAR2) return VARCHAR2;
3.        FUNCTION  JavaFunction1 return VARCHAR2;
4.        PROCEDURE JavaProcedure(userName IN VARCHAR2);
5.        PROCEDURE JavaTrigger(deptNo IN NUMBER,oldName IN VARCHAR2,newName IN VARCHAR2);
6.    END HelloJSP ;
7.    /

2)

01.CREATE OR REPLACE PACKAGE BODY HelloJSP as
02.        FUNCTION JavaFunction(userName IN VARCHAR2)
03.           RETURN VARCHAR2
04.           AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction(java.lang.String) return java.lang.String';
05. 
06.        FUNCTION JavaFunction1
07.           RETURN VARCHAR2
08.           AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction1() return java.lang.String';
09. 
10.        PROCEDURE JavaProcedure(userName IN VARCHAR2)
11.          AS LANGUAGE JAVA NAME 'HelloJSP.javaProcedure(java.lang.String)';
12. 
13.        PROCEDURE javaTrigger(deptNo IN NUMBER, oldName IN VARCHAR2, newName IN VARCHAR2)
14.           AS LANGUAGE JAVA NAME 'HelloJSP.javaTrigger(int, java.lang.String, java.lang.String)';
15.    end HelloJSP ;
16.    /

创建包的主体时,定义函数或过程体时不能加 "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.CREATE OR REPLACE PROCEDURE JavaTrigger(deptNo IN NUMBER, oldName IN VARCHAR2, newName IN VARCHAR2)
2.       AUTHID CURRENT_USER
3.       AS LANGUAGE JAVA NAME 'HelloJSP.javaTrigger(int, java.lang.String, java.lang.String)';
4.   /

2)
1.CREATE OR REPLACE TRIGGER CallJavaTrigger
2.      AFTER update OR delete OR insert ON dept
3.      FOR EACH ROW
4.      call JavaTrigger(:old.deptno,:old.dname,:new.dname)
5.   /

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官方文档)

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/673037/viewspace-624879/,如需转载,请注明出处,否则将追究法律责任。

上一篇: WinXP
下一篇: Oracle库的Hang
请登录后发表评论 登录
全部评论

注册时间:2008-07-25

  • 博文量
    9
  • 访问量
    20648