Java调用mysql的存储过程

MySQL5存储过程,用得人好像不多。按照数据库设计原理来讲,存储过程是在db server上预编译的,所以查询速度会比较起纯SQL语句快很多。可能是现在流行OO,导至存储过程使用的余地大打折扣。但如果从效果上来讲,用存储过程来实现业务规则所带得DB SERVER压力,比用JAVA类实现业务规则所带来的WebServer压力要小。当然存储过程也不应滥用,象普通的insert、update之类的语句就不需要使用存储过程了。

存储过程的作用

数据库存储过程:简单滴说,存储过程就是存储在数据库中的一个程序。

第一:存储过程因为SQL语句已经预编绎过了,因此运行的速度比较快。   
第二:存储过程可以接受参数、输出参数、返回单个或多个结果集以及返回值。可以向程序返回错误原因。       
第三:存储过程运行比较稳定,不会有太多的错误。只要一次成功,以后都会按这个程序运行。       
第四:存储过程主要是在服务器上运行,减少对客户机的压力。     
第五:存储过程可以包含程序流、逻辑以及对数据库的查询。同时可以实体封装和隐藏了数据逻辑。     
第六:存储过程可以在单个存储过程中执行一系列SQL语句。     
第七:存储过程可以从自己的存储过程内引用其它存储过程,这可以简化一系列复杂语句。  

此外,如果多条SQL语句执行过程中,过程环节返回了数据作为后面环节的输入数据,如果直接通过SQL语句执行,势必导致大量的数据通过网络返回到客户机,并在客户机运算;如果封装在存储过程中,则将运算放在服务器进行,不但减少了客户机的压力,同时也减少了网络流量,提高了执行的效率。

书写procedure语句

言归正传,MySQL创建procudure的语法与sql server/Oracle差别较大。

例1:传入参数的procedure

create procedure usp_test(param varchar(20)
  select * from talbeName where column=param 

例2:更新表的procedure

create procedure usp_test2 (t varchar(20))
  begin
    set xname = 'test';
    update table set column = xname where column1=t;
  end

MySQL创建存储过程时不带as,而且()不能省略,即便是没有传入参数。这一点象sql server或是oracle那样直观,而且声明变量时直接用declare,不用加的@或@@(in out变量除外)。还有一点是很怪的语法,如果是以“select”为开头的存储过程,是不能加”begin end”的。”begin end”表示多条SQL语句的复合体

当然,事实上几乎没有哪款数据库是完全符合SQL3标准的,多少都含有自己的一些成份里面,这也造成使用存储过程会使程序的可移性降低。

调用procedure方法

MySQL使用call关键字。例:call usp_test(‘test’);而不是execute,同样()是不能省略的。
创建完procedure后,再看看java是如何调用procedure的。

Connection conn = null;
CallableStatement cstmt = null;
ResultSet rs = null;
try {
    conn = DbConn.getDbConn();// get pool conn
    CallableStatement cstmt = conn.prepareCall("{call usp_test(?)}");
    call.setString(1, "test");
    rs = call.executeQuery();
    while (rs.next()) {
        String te = rs.getString(1);
        System.out.println("te:" + te);
    }
} catch (Exception e) {
    System.out.println("e: " + e);
} finally {
    try {
        rs.close();
        cstmt.close();
        conn.close();
    } catch (Exception ex) {
        System.out.println("ex:" + ex);
    }
}

这里使用的是”{call usp_test()}”来调用存储过程。同时也可以编程传入参数,进行查询。
上述方法有个致命的缺点,就是传入的参数是由编写存储过程决定的。也就是说不支持象PreparedStatement那样的多参数动态查询。

下面先讲解MySQL存储过程的输入输出参数应用。MySQL调用存储过程的语法总结一下就是三个参数关键值IN、OUT、INOUT,分别代表输入参数、输出参数、输入输出参数。

输入参数(in)

存储过程sql:

CREATE PROCEDURE usp1(IN p INT)
BEGIN
    SET @x = pEND;

sql调用方法:

call usp1(123456);
 select @x

java调用方法:

CallableStatement callStatement = conn.prepareCall("{call usp1(?)}"); // 调用存储过程
callStatement.setInt(1,123456); // 设置参数
callStatement.execute(); // 执行
conn.close(); // 关闭连接

返回结果:

返回结果为 123456

输出参数(out)

存储过程sql:

CREATE PROCEDURE usp2 (OUT p int,)
BEGIN
    SET p = 5 ;
END

sql调用方法:

call p6(@y)  // @y表示输出,10表示输入
select @y

java调用方法:

CallableStatement callStatement = conn.prepareCall("{call usp2(?)}"); // 调用存储过程
// 注册存储过程的out型参数类型,使用之前必须要先注册
callStatement.registerOutParameter(1,java.sql.Types.INTEGER);    
callStatement.execute(); // 执行
System.out.println(callStatement.getInt(1)); //获取存储过程的输出参数
conn.close(); // 关闭连接

返回结果:

返回结果为 5

输入输出参数(in out)

存储过程sql:

CREATE PROCEDURE demoSp(IN inputParam VARCHAR(255), INOUT inOutParam INT)
BEGIN
    SET inOutParam = 1000;
    SELECT inOutParam;
    SELECT CONCAT('zyxw ', inputParam);
END

sql调用方法:

call demoSP(' test',@q)

java调用方法:

CallableStatement callStatement = conn.prepareCall("{call demoSp(?,?)}"); // 调用存储过程
// 注册存储过程的out型参数类型,使用之前必须要先注册
callStatement.registerOutParameter(1,java.sql.Types.STRING);
callStatement.setString(" test"); //设置参数    
boolean flag = callStatement.execute(); // 执行
System.out.println(callStatement.getString(1)); //获取存储过程的输出参数
conn.close(); // 关闭连接

返回结果:

返回结果"zyxw  test"

与没有输出参数的存储过程相比,多一个registerOutParameter。也就是说要先注册输出参数的类型,再执行execute(),最后取回车出参数的值。这个过程是不能颠倒的。

参考

java调用mysql的存储过程(简单示例)
java 调用MySQL存储过程(转载)