ITPub博客

首页 > Linux操作系统 > Linux操作系统 > Oracle8i中使用Java语言来开发存储过程

Oracle8i中使用Java语言来开发存储过程

原创 Linux操作系统 作者:zhanglincon 时间:2009-04-30 14:36:20 0 删除 编辑

. 存储过程简介

存储过程是存储在数据库中的一段存储程序。当创建存储过程时,系统会对其进行编译,并将执行代码存储到数据库中。

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. import java.io.*;   
  2. import java.net.*;   
  3. import java.sql.*;   
  4.   
  5. import oracle.jdbc.driver.OracleDriver;   
  6.   
  7. public class HelloJSP {   
  8.   
  9.   
  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 os = 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. }  
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 s = 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. CREATE OR REPLACE PROCEDURE JavaProcedure(userName IN VARCHAR2)   
  2.        AUTHID CURRENT_USER  
  3.        AS LANGUAGE JAVA NAME 'HelloJSP.javaProcedure(java.lang.String)';   
  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. 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.    /  
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. CREATE OR REPLACE FUNCTION JavaFunction1    
  2.       RETURN VARCHAR2   
  3.       AUTHID CURRENT_USER  
  4.       AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction1() return java.lang.String';   
  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. 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.     /  
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. CREATE OR REPLACE PACKAGE BODY HelloJSP as  
  2.         FUNCTION JavaFunction(userName IN VARCHAR2)    
  3.            RETURN VARCHAR2   
  4.            AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction(java.lang.String) return java.lang.String';   
  5.   
  6.         FUNCTION JavaFunction1    
  7.            RETURN VARCHAR2   
  8.            AS LANGUAGE JAVA NAME 'HelloJSP.javaFunction1() return java.lang.String';   
  9.   
  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.     /  
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. 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.    /  
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. 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.    /  
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中就是不可见了 

 

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

上一篇: 一条shell命令
请登录后发表评论 登录
全部评论

注册时间:2009-03-24

  • 博文量
    79
  • 访问量
    206036