JDBC Statement, PreparedStatement 和 CallableStatement

jdbc statement, preparedstatement 和 callablestatement

獲得了數(shù)據(jù)庫連接后,我們就可以與數(shù)據(jù)庫進行交互了。jdbc 中的 statement, callablestatement 和 preparedstatement 三個對象,定義了一系列的方法和屬性,可以用來向數(shù)據(jù)庫發(fā)送 sql 命令,接收數(shù)據(jù)。

接口 使用說明
statement 運行靜態(tài) sql 語句時使用,statement 不能接收輸入?yún)?shù)。
preparedstatement 當(dāng)計劃多次使用某條 sql 語句時使用,并且 preparedstatement 可以接收輸入?yún)?shù)。
callablestatement 用來調(diào)用數(shù)據(jù)庫的存儲過程,callablestatement 可以接收輸入?yún)?shù)。

 

1. statement 對象

1)創(chuàng)建 statement 對象

statement 對象由 connection 對象的 createstatement( ) 方法創(chuàng)建,用來執(zhí)行靜態(tài) sql 語句,如下所示:

statement stmt = null;
try {
   stmt = conn.createstatement( );
   . . .
}
catch (sqlexception e) {
   . . .
}
finally {
   . . .
}

一旦創(chuàng)建了一個 statement 對象,然后可以它的三個執(zhí)行方法之一執(zhí)行 sql 語句。

  • boolean execute(string sql) :這個方法常用來執(zhí)行 sql ddl 語句。
  • int executeupdate(string sql) :返回 sql 語句執(zhí)行時受影響的行數(shù),常用來執(zhí)行 insert,update 或 delete 語句。
  • resultset executequery(string sql) : 返回 resultset 對象。常用來執(zhí)行 select 語句,會得到一個結(jié)果集 resultset。

2)關(guān)閉 statement 對象

正如關(guān)閉一個 connection 對象來釋放數(shù)據(jù)庫資源,出于同樣的原因,也應(yīng)該關(guān)閉 statement 對象。

使用 close() 方法關(guān)閉 statement。

statement stmt = null;
try {
   stmt = conn.createstatement( );
   . . .
}
catch (sqlexception e) {
   . . .
}
finally {
   stmt.close();
}

 

2. preparedstatement 對象

preparedstatement 接口擴展了 statement 接口,給 statement 對象增加幾個高級功能。

它對 sql 語句進行預(yù)編譯,效率更高。另外,可以接收動態(tài)參數(shù),避免 statement 中的 sql 注入問題。

1)創(chuàng)建 preparedstatement 對象

preparedstatement pstmt = null;
try {
   string sql = "update employees set age = ? where id = ?";
   pstmt = conn.preparestatement(sql);
   pstmt.setint(1, 22);
   . . .
}
catch (sqlexception e) {
   . . .
}
finally {
   . . .
}

在 jdbc 中參數(shù)使用 ?代表,在執(zhí)行 sql 語句之前,必須提供每一個參數(shù)的值。

setxxx() 方法將值綁定到參數(shù),其中 xxx 表示希望綁定到輸入?yún)?shù)值的 java 數(shù)據(jù)類型。如果忘了提供值,將收到一個sqlexception。

每個參數(shù)標(biāo)記是由它的序號位置引用。第一標(biāo)記表示位置 1,下一個位置為 2 等等。這種方法不同于 java 數(shù)組索引,以 0 開始。

2)關(guān)閉 preparedstatement 對象

正如關(guān)閉一個 connection 對象來釋放數(shù)據(jù)庫資源,出于同樣的原因,也應(yīng)該關(guān)閉 preparedstatement 對象。

使用 close() 方法關(guān)閉 preparedstatement。

preparedstatement pstmt = null;
try {
   string sql = "update employees set age = ? where id = ?";
   pstmt = conn.preparestatement(sql);
   . . .
}
catch (sqlexception e) {
   . . .
}
finally {
   pstmt.close();
}

 

3. callablestatement 對象

正如一個connection對象創(chuàng)建statement和preparedstatement對象,它也創(chuàng)造了callablestatement對象這將被用來執(zhí)行調(diào)用數(shù)據(jù)庫存儲過程。

1)創(chuàng)建 callablestatement 對象

假設(shè)需要執(zhí)行以下 oracle 存儲過程:

create or replace procedure getempname 
   (emp_id in number, emp_first out varchar) as
begin
   select first into emp_first
   from employees
   where id = emp_id;
end;

存儲過程有三種類型的參數(shù):in,out和inout。 preparedstatement對象只使用in參數(shù)。 callablestatement對象可以使用所有三個。

這里是每個的定義:

參數(shù) 描述
in 它的值是在創(chuàng)建sql語句時未知的參數(shù)。將值綁定到與 setxxx() 方法的參數(shù)。
out 其值由它返回的sql語句提供的參數(shù)。從 out 參數(shù)的 getxxx() 方法檢索值。
inout 同時提供輸入和輸出值的參數(shù)。綁定變量使用 setxxx() 方法,使用 getxxx() 方法檢索值。

下面的代碼顯示了使用 connection.preparecall() 方法獲得存儲過程 callablestatement 對象:

callablestatement cstmt = null;
try {
   string sql = "{call getempname (?, ?)}";
   cstmt = conn.preparecall (sql);
   . . .
}
catch (sqlexception e) {
   . . .
}
finally {
   . . .
}

sql 表示存儲過程,里面使用了參數(shù)占位符。

使用 callablestatement 對象是就像使用 preparedstatement 對象,執(zhí)行該語句之前,必須將值綁定到所有的參數(shù),否則將收到一個 sqlexception。

如果有 in 參數(shù),只要按照 preparedstatement 對象相同的規(guī)則,使用 setxxx() 方法綁定的 java 數(shù)據(jù)類型。

當(dāng)使用 out 和 inout 參數(shù)就必須采用額外的 callablestatement 及 registeroutparameter() 方法。registeroutparameter() 方法綁定數(shù)據(jù)類型到存儲過的返回值。

2)關(guān)閉 callablestatement 對象

正如關(guān)閉其他 statement 對象,出于同樣的原因,也應(yīng)該關(guān)閉 callablestatement 對象。

正如關(guān)閉一個 connection 對象來釋放數(shù)據(jù)庫資源,出于同樣的原因,也應(yīng)該關(guān)閉 callablestatementv 對象。

使用 close() 方法關(guān)閉 callablestatement。

callablestatement cstmt = null;
try {
   string sql = "{call getempname (?, ?)}";
   cstmt = conn.preparecall (sql);
   . . .
}
catch (sqlexception e) {
   . . .
}
finally {
   cstmt.close();
}
相關(guān)文章
亚洲国产精品第一区二区,久久免费视频77,99V久久综合狠狠综合久久,国产免费久久九九免费视频