自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

淺談JDBC DAO的設(shè)計理念

開發(fā) 后端
本文將簡單談?wù)凧DBC DAO的設(shè)計理念,DAO是Data Access Object數(shù)據(jù)訪問接口,數(shù)據(jù)訪問:顧名思義就是與數(shù)據(jù)庫打交道。

JDBC DAO中Connection的含義

Connection表示了一個和數(shù)據(jù)庫的鏈接,底層需要有操作系統(tǒng)的Socket支持,所以Connection是一種資源,既然是一種資源,就需要按照建立,打開,使用,關(guān)閉的順序合理的使用。

Connection是Java數(shù)據(jù)庫操作的基礎(chǔ),是進行一系列操作的基礎(chǔ),所有的派生的操作,例如Statement,PreparedStatement,ResultSet等都由Connection直接或者間接的衍生。

如何獲得Connection呢?

方法一,使用DriverManager類來獲取,前提條件是數(shù)據(jù)庫驅(qū)動程序需要在classpath下(即使用數(shù)據(jù)庫鏈接的程序按照Java的方式可以訪問到)。

Connectionconn=DriverManager.getConnection("jdbc:oracle:thin:@192.168.0.1:1521:ORCL",user,pwd);

方法二,使用數(shù)據(jù)庫連接池來獲取

什么是數(shù)據(jù)庫連接池呢,數(shù)據(jù)庫連接池是標(biāo)準(zhǔn)JavaEE容器的一種服務(wù),例如Webspher,Weblogic,Tomcat等,容器預(yù)先建立一些數(shù)據(jù)庫鏈接,以便應(yīng)用程序使用的時候從中借取,注意有借有還,當(dāng)應(yīng)用程序使用完了之后會將數(shù)據(jù)庫鏈接還回連接池。(數(shù)據(jù)源配置請參考其他文檔)

使用連接池的好處是,可以預(yù)先建立鏈接,減小在數(shù)據(jù)庫獲取上的相對時間。

使用連接池獲取數(shù)據(jù)庫鏈接的方式為:

  1. InitialContextctx=newInitialContext();
  2. DataSourceds=(DataSource)ctx.lookup("java:comp/env/jdbc/DataSource");
  3. Connectionconn=ds.getConnection();

由于在配置數(shù)據(jù)庫連接池的時候已經(jīng)定義了URL,用戶名,密碼等信息,所以在程序中使用的時候不需要傳入這些信息。

JDBC DAO中ConnectionManager定義

Connection用來專門管理數(shù)據(jù)庫鏈接,通常情況下ConnectionManager只有一個方法,調(diào)用這個方法將返回一個Connection的實例。通過ConnectionManager可以封裝Connection的獲取方式(例如開發(fā)的時候使用DriverManager,運用的時候使用DataSource的方式,但是不需要修改ConnectionManager之外的其他代碼)和追加Connection獲取之前之后的操作(例如針對Connection的屬性的設(shè)置)。

下面的代碼是一個ConnectionManager的代碼示例:

  1. packagecom.jpleasure.jdbc.dao;
  2. importjava.sql.Connection;
  3. importjava.sql.DriverManager;
  4. importjava.sql.SQLException;
  5. publicclassConnectionManager{
  6. publicstaticConnectiongetConnection()throwsDaoException{
  7. Connectionconn=null;
  8. try{
  9. conn=DriverManager.getConnection("","","");
  10. }catch(SQLExceptione){
  11. thrownewDaoException("cannotgetdatabaseconnection",e);
  12. }
  13. returnconn;
  14. }
  15. }

如果需要從開發(fā)模式變?yōu)檫\用模式,只需要將上述代碼修改為:

  1. packagecom.jpleasure.jdbc.dao;
  2. importjava.sql.Connection;
  3. importjava.sql.DriverManager;
  4. importjava.sql.SQLException;
  5. publicclassConnectionManager{
  6. publicstaticConnectiongetConnection()throwsDaoException{
  7. Connectionconn=null;
  8. try{
  9. Contextctx=newInitialContext();
  10. DataSourceds=(DataSource)ctx.lookup("jdbc/dsname");
  11. conn=ds.getConnection();
  12. }catch(NamingExceptione){
  13. thrownewDaoException("cannotfinddatasource",e);
  14. }catch(SQLExceptione){
  15. thrownewDaoException("cannotgetdatabaseconnection",e);
  16. }
  17. returnconn;
  18. }
  19. }
  20. 如果需要預(yù)先設(shè)定Connection的一些屬性,也可以在上述代碼中設(shè)定,例如:
  21. packagecom.jpleasure.jdbc.dao;
  22. importjava.sql.Connection;
  23. importjava.sql.DriverManager;
  24. importjava.sql.SQLException;
  25. publicclassConnectionManager{
  26. publicstaticConnectiongetConnection()throwsDaoException{
  27. Connectionconn=null;
  28. try{
  29. Contextctx=newInitialContext();
  30. DataSourceds=(DataSource)ctx.lookup("jdbc/dsname");
  31. conn=ds.getConnection();
  32. conn.setAutoCommit(false);
  33. }catch(NamingExceptione){
  34. thrownewDaoException("cannotfinddatasource",e);
  35. }catch(SQLExceptione){
  36. thrownewDaoException("cannotgetdatabaseconnection",e);
  37. }
  38. returnconn;
  39. }
  40. }
  41. CommonDao定義
  42. 屬性和構(gòu)造方法
  43. 通常情況下,CommonDao要有一個Connection的引用。所有一個CommonDao的實例的所有方法的調(diào)用都需要依賴于這個Connection。需要一個Connection的另外一個原因是如果各個方法需要保證在一個事務(wù)環(huán)境中(上下文中),必須保證所有的操作都在一個Connection上。
  44. 構(gòu)造方法通常需要將類型為Connection的屬性實例化,例如:
  45. packagecom.jpleasure.jdbc.dao;
  46. importjava.sql.Connection;
  47. publicclassCommonDao{
  48. privateConnectionconn;
  49. publicCommonDao()throwsDaoException{
  50. this.conn=ConnectionManager.getConnection();
  51. }
  52. }

事務(wù)方法

begin()

開始一個事務(wù),調(diào)用CommonDao的begin方法之后,所以的后續(xù)操作將會在一個事務(wù)環(huán)境內(nèi),要么全部提交,要么全部回滾。

commit()

提交一個事務(wù),必須在begin調(diào)用之后調(diào)用。且和rollback方法互斥。

rollback()

回滾一個事務(wù),必須在begin方法調(diào)用之后調(diào)用。且和commit方法互斥。

事務(wù)的實現(xiàn)有兩種方法,一種是使用基于單一Connection的事務(wù),另外一種方法是使用容器的JTA(JavaTransactionAPI)。需要注意的是***種方法可以在任何環(huán)境下使用,但是只能是針對單一的數(shù)據(jù)庫鏈接。第二種方法智能在支持JTA的JavaEE容器中使用(例如Websphere,Weblogic等,Tomcat默認不支持),但是支持多個Connection實例。

***種方法代碼為:

  1. packagecom.jpleasure.jdbc.dao;
  2. importjava.sql.Connection;
  3. importjava.sql.SQLException;
  4. publicclassCommonDao{
  5. privateConnectionconn;
  6. publicCommonDao()throwsDaoException{
  7. this.conn=ConnectionManager.getConnection();
  8. }
  9. publicvoidbegin()throwsDaoException{
  10. if(conn!=null){
  11. try{
  12. conn.setAutoCommit(false);
  13. }catch(SQLExceptione){
  14. thrownewDaoException("cannotbegintransaction",e);
  15. }
  16. }else{
  17. thrownewDaoException("connectionnotopened!");
  18. }
  19. }
  20. publicvoidcommit()throwsDaoException{
  21. try{
  22. if(conn!=null&&!conn.getAutoCommit()){
  23. conn.commit();
  24. conn.setAutoCommit(true);
  25. }else{
  26. if(conn==null){
  27. thrownewDaoException("connectionnotopened!");
  28. }else{
  29. thrownewDaoException("firstbeginthencommitplease!");
  30. }
  31. }
  32. }catch(SQLExceptione){
  33. thrownewDaoException("cannotcommittransaction!",e);
  34. }
  35. }
  36. publicvoidrollback()throwsDaoException{
  37. try{
  38. if(conn!=null&&!conn.getAutoCommit()){
  39. conn.rollback();
  40. conn.setAutoCommit(true);
  41. }else{
  42. if(conn==null){
  43. thrownewDaoException("connectionnotopened!");
  44. }else{
  45. thrownewDaoException("firstbeginthenrollbackplease!");
  46. }
  47. }
  48. }catch(SQLExceptione){
  49. thrownewDaoException("cannotrollbacktransaction!",e);
  50. }
  51. }
  52. }

第二種我們在使用DAO的實例中介紹如何使用(@TODO)

新建兩個DAO,做不同的操作,使用JTA保證事務(wù)完整。

查詢方法

查詢方法也許是CommonDao最常用的方法,查詢方法需要將數(shù)據(jù)庫的結(jié)果返回給畫面。返回值我們一般不使用ResultSet,因為ResultSet依賴于Connection,如果Connection關(guān)閉,ResultSet將不再有效,所以我們通常將ResultSet轉(zhuǎn)變?yōu)橐粋€List之后返回。

在說明查詢方法之前,我們先說說如何將數(shù)據(jù)庫中的內(nèi)容放在List中,我們使用一個List表示一個查詢結(jié)果集合,使用一個Map表示集合中的一行,Map的key表示數(shù)據(jù)庫表的字段名字,Value表示數(shù)據(jù)庫字段的內(nèi)容。代碼為:

  1. privateListconvert(ResultSetrs)throwsDaoException{
  2. //recordlist
  3. ListretList=newArrayList();
  4. try{
  5. ResultSetMetaDatameta=rs.getMetaData();
  6. //columncount
  7. intcolCount=meta.getColumnCount();
  8. //eachrecord
  9. while(rs.next()){
  10. MaprecordMap=newHashMap();
  11. //eachcolumn
  12. for(inti=1;i<=colCount;i++){
  13. //columnname
  14. Stringname=meta.getColumnName(i);
  15. //columnvalue
  16. Objectvalue=rs.getObject(i);
  17. //addcolumntorecord
  18. recordMap.put(name,value);
  19. }
  20. //adrecordtolist
  21. retList.add(recordMap);
  22. }
  23. }catch(SQLExceptionex){
  24. thrownewDaoException("cannotconvertresultsettolistofmap",ex);
  25. }
  26. returnretList;
  27. }

為了避免Sql注入的安全問題,我們通常使用PreparedStatement,在使用PreparedStatement的時候涉及到如何將傳入?yún)?shù)設(shè)置到PreparedStatement上面,參看以下的共通方法:

  1. privatevoidapply(PreparedStatementpstmt,Listparams)throwsDaoException{
  2. try{
  3. //ifparamsexist
  4. if(params!=null&¶ms.size()>0){
  5. //parametersiterator
  6. Iteratorit=params.iterator();
  7. //parameterindex
  8. intindex=1;
  9. while(it.hasNext()){
  10. Objectobj=it.next();
  11. //ifnullset""
  12. if(obj==null){
  13. pstmt.setObject(index,"");
  14. }else{
  15. //elsesetobject
  16. pstmt.setObject(index,obj);
  17. }
  18. //nextindex
  19. index++;
  20. }
  21. }
  22. }catch(SQLExceptionex){
  23. thrownewDaoException("cannotapplyparameter",ex);
  24. }
  25. }
  26. 接著我們繼續(xù)說我們的查詢方法,有了上述兩個方法,我們的查詢方法就非常簡單了:
  27. publicListquery(Stringsql,Listparams)throwsDaoException{
  28. Listresult=null;
  29. PreparedStatementpstmt=null;
  30. ResultSetrs=null;
  31. try{
  32. pstmt=conn.prepareStatement(sql);
  33. this.apply(pstmt,params);
  34. rs=pstmt.executeQuery();
  35. result=this.convert(rs);
  36. }catch(SQLExceptionex){
  37. thrownewDaoException("cannotexecutequery",ex);
  38. }finally{
  39. if(rs!=null){
  40. try{
  41. rs.close();
  42. }catch(SQLExceptione){
  43. //nothing
  44. }
  45. }
  46. if(pstmt!=null){
  47. try{
  48. pstmt.close();
  49. }catch(SQLExceptione){
  50. //nothing
  51. }
  52. }
  53. }
  54. returnresult;
  55. }

特殊的查詢方法(返回單值)

有時候為了方便使用,我們需要返回單值的產(chǎn)尋方法,例如selectmax(id)fromtable_a,selectcount(id)fromtable_b等。以下的代碼使用了上述通用的查詢方法,代碼為:

  1. publicObjectqueryOne(Stringsql,Listparams)throwsDaoException{
  2. Listlist=this.query(sql,params);
  3. if(list==null||list.size()==0){
  4. thrownewDaoException("datanotexist");
  5. }else{
  6. Maprecord=(Map)list.get(0);
  7. if(record==null||record.size()==0){
  8. thrownewDaoException("datanotexist");
  9. }else{
  10. returnrecord.values().toArray()[0];
  11. }
  12. }
  13. }

更新,刪除,插入方法

由于在JDBC中這三個方法都是用了一個execute完成,所以這里我們也使用一個方法來完成這些功能。代碼為:

  1. publicintexecute(Stringsql,Listparams)throwsDaoException{
  2. intret=0;
  3. PreparedStatementpstmt=null;
  4. try{
  5. pstmt=conn.prepareStatement(sql);
  6. this.apply(pstmt,params);
  7. ret=pstmt.executeUpdate();
  8. }catch(SQLExceptionex){
  9. thrownewDaoException("",ex);
  10. }finally{
  11. if(pstmt!=null){
  12. try{
  13. pstmt.close();
  14. }catch(SQLExceptione){
  15. //nothing.
  16. }
  17. }
  18. }
  19. returnret;
  20. }

批處理方法(查詢)

有些時候為了便于操作,需要一次查詢多條SQL語句,我們稱之為批處理,實現(xiàn)參看以下方法,其中為了和query方法做區(qū)分,將參數(shù)和返回值都改為了數(shù)組形式。

  1. publicList[]queryBatch(String[]sqlArray,List[]paramArray)throwsDaoException{
  2. Listrets=newArrayList();
  3. if(sqlArray.length!=paramArray.length){
  4. thrownewDaoException("sqlsizenotequalparametersize");
  5. }else{
  6. for(inti=0;iStringsql=sqlArray[i];
  7. Listparam=paramArray[i];
  8. Listret=this.query(sql,param);
  9. rets.add(ret);
  10. }
  11. return(List[])rets.toArray();
  12. }
  13. }

批處理方法(更新)

有些時候需要一次更新多條Sql語句,為了便于操作,添加了批處理更新操作,參看以下代碼,為了和更新方法區(qū)分,將參數(shù)和返回值都改為了數(shù)組形式。

  1. publicint[]executeBatch(String[]sqlArray,List[]paramArray)throwsDaoException{
  2. Listrets=newArrayList();
  3. if(sqlArray.length!=paramArray.length){
  4. thrownewDaoException("sqlsizenotequalparametersize");
  5. }else{
  6. for(inti=0;iintret=this.execute(sqlArray[i],paramArray[i]);
  7. rets.add(newInteger(ret));
  8. }
  9. int[]retArray=newint[rets.size()];
  10. for(inti=0;iretArray[i]=((Integer)rets.get(i)).intValue();
  11. }
  12. returnretArray;
  13. }
  14. }

資源釋放

由于CommonDao有一個Connection的屬性,且Connection屬于稀缺資源,所以在CommonDao不需要在使用的時候需要顯示的關(guān)閉Connection。代碼如下:

  1. publicvoidclose()throwsDaoException{
  2. try{
  3. if(conn!=null&&conn.getAutoCommit()){
  4. conn.close();
  5. }else{
  6. if(conn==null){
  7. thrownewDaoException("cannotclosenullconnection,firstnewthenclose");
  8. }else{
  9. thrownewDaoException("transactionisrunning,rollbakcorcommitbeforcloseplease.");
  10. }
  11. }
  12. }catch(SQLExceptionex){
  13. thrownewDaoException("Cannotclosecommondao");
  14. }
  15. }

JDBC工具類(JDBCUtilClass)

在上述的代碼中我們看到有很多的無用的處理,例如:

  1. if(pstmt!=null){
  2. try{
  3. pstmt.close();
  4. }catch(SQLExceptione){
  5. //nothing.
  6. }
  7. }

為什么要有這些處理呢?說先這些處理發(fā)生的位置都是在正常處理完成之后,這些處理(例如pstmt.close())即使失敗也沒有影響,這個時候我們需要做上述的無用處理,這正是JDBCAPI的一個小小的瑕疵。我們通常使用一個特殊的靜態(tài)工具來來做補充,例如:

  1. packagecom.jpleasure.jdbc.dao;
  2. importjava.sql.Connection;
  3. importjava.sql.PreparedStatement;
  4. importjava.sql.ResultSet;
  5. importjava.sql.SQLException;
  6. publicclassJDBCUtil{
  7. publicvoidsafelyClose(Connectionconn){
  8. if(conn!=null){
  9. try{
  10. conn.close();
  11. }catch(SQLExceptione){
  12. //
  13. }
  14. }
  15. }
  16. publicvoidsafelyClose(PreparedStatementpstmt){
  17. if(pstmt!=null){
  18. try{
  19. pstmt.close();
  20. }catch(SQLExceptione){
  21. //
  22. }
  23. }
  24. }
  25. publicvoidsafelyClose(ResultSetrs){
  26. if(rs!=null){
  27. try{
  28. rs.close();
  29. }catch(SQLExceptione){
  30. //
  31. }
  32. }
  33. }
  34. }

JDBC DAO中異常處理

也許細心的你已經(jīng)發(fā)現(xiàn)了一個問題,為什么所有拋出異常的地方我們都是將SQLException包裝在了DaoException之內(nèi)拋出呢,為什么不直接拋出SQLException呢?有兩個原因,***,可以細化,分類Exception拋出合適的異常,添加合適的消息,第二,隔離和Dao和業(yè)務(wù)邏輯的耦合,可以方便的修改Dao層而不會影響到業(yè)務(wù)邏輯層。另外需要注意,DaoExcetion中可以包含SQLException,這個時候可以為客戶提供更詳細的錯誤信息,例如ORA-12524等內(nèi)容,但是很少見到。

  1. packagecom.jpleasure.jdbc.dao;
  2. publicclassDaoExceptionextendsException{
  3. publicDaoException(){
  4. super();
  5. }
  6. publicDaoException(Stringmessage,Throwablecause){
  7. super(message,cause);
  8. }
  9. publicDaoException(Stringmessage){
  10. super(message);
  11. }
  12. publicDaoException(Throwablecause){
  13. super(cause);
  14. }
  15. }

【編輯推薦】

  1. 使用JDBC的五個精華功能
  2. Tomcat5+MySQL JDBC連接池配置
  3. 在Weblogic中實現(xiàn)JDBC的功能
  4. 詳解JDBC與Hibernate區(qū)別
  5. JDBC連接MySQL數(shù)據(jù)庫關(guān)鍵四步
  6. 詳解JDBC驅(qū)動的四種類型
責(zé)任編輯:彭凡 來源: javaeye
相關(guān)推薦

2009-06-29 17:17:57

Spring

2009-07-15 17:11:31

JDBC的概念

2009-07-20 17:41:59

Java JDBC

2009-07-01 17:58:20

JSP

2009-07-21 17:41:58

JDBC數(shù)據(jù)源

2010-03-02 16:34:06

Android平臺

2009-07-15 18:07:47

JDBC代碼

2009-07-20 09:27:42

IBATIS.netDAO

2010-03-16 17:07:51

云計算

2010-09-28 11:05:49

jQuery

2022-10-09 14:15:42

短鏈設(shè)計

2009-07-23 13:37:45

JDBC連接SQL S

2009-07-16 14:46:48

jdbc statem

2010-11-09 09:43:22

UI設(shè)計Windows Pho

2009-07-17 17:41:25

JDBC連接SQL S

2010-06-11 14:55:20

2009-07-22 13:49:40

JSP JDBC

2009-07-16 16:23:20

JDBC result

2009-07-15 15:30:12

MyEclipse J

2009-07-14 17:18:23

JDBC怎么連接數(shù)據(jù)庫
點贊
收藏

51CTO技術(shù)棧公眾號