JSP頁面查詢顯示常用模式 (附原始碼)

來源:互聯網
上載者:User
JSP頁面查詢顯示常用模式 title: JSP頁面查詢顯示常用模式

author: evan

email: evan_zhao@hotmail.com
    
背景
1.    需要將資料庫查詢結果在JSP中以列表方式顯示
2.    在一個良好的J2EE模式中資料庫查詢一般用DAO實現(Data Access Object), JSP僅用於顯示資料

問題
    通過JDBC ResultSet可擷取查詢結果(存在於資料庫緩衝區內),但在Statement、Connection關閉後ResultSet即不可用。因此需要一種方式取出所有查詢結果並傳遞至JSP頁面。

解決方案一
    使用Value Object。將每條記錄均封裝成JavaBean對象,把這些對象裝入Collection傳送給JSP顯示。這種方法的缺點是每一種查詢都需要定義一個java class,並且將記錄資料封裝成java對象時也需要很多額外的代碼。
範例程式碼:


  1. //查詢資料代碼
  2.   Connection conn = DBUtil.getConnection();
  3.   PreparedStatement pst = null;
  4.   ResultSet rs = null;
  5.   try{
  6.     String sql=“select emp_code, real_name from t_employee where organ_id=?”;
  7.     pst = conn.preparedStatement(sql);
  8.     pst.setString(1, “101”);
  9.     ResultSet rs = pst.executeQuery();
  10.     List list = new ArrayList();
  11.     Employee emp;
  12.     while (rs.next()){
  13.       emp = new Employee();
  14.       emp.setReakName(rs.getString(“real_name”));
  15.       emp.setEmpCode(rs.getString(“emp_code”));
  16.       …
  17.       list.add(emp);
  18.     }
  19.     return list;
  20.   }finally{
  21.     DBUtil.close(rs, pst ,conn);
  22.   }


  23. //jsp顯示部分代碼
  24. <%
  25.   List empList = (List)request.getAttribute(“empList”);
  26.   if (empList == null) empList = Collections.EMPTY_LIST;
  27. %>
  28. <table  cellspacing="0" width=”90%”>
  29.     <tr>  <td>代碼</td> <td>姓名</td>  </tr>
  30. <%
  31.   Employee emp;
  32.   for (int i=0; i< empList.size(); i++){
  33.     emp = (Employee) empList.get(i);
  34. %>
  35.     <tr>  
  36.       <td><%= emp.getEmpCode()%></td> 
  37.       <td><%= emp.getRealName()%></td>  
  38.     </tr>
  39. <%
  40.   }// end for
  41. %>
  42. </table>

解決方案二
    遍曆ResultSet取出所有資料封裝進Collection。
具體做法:
1.    產生一個List對象(List list = new ArrayList() )。
2.    產生一個Map對象(Map map = new HashMap() )。使用Map封裝一行資料,key為各欄位名,value為對應的值。(map.put(“USER_NAME”), rs.getString(“USER_NAME”))
3.    將第2 步產生的Map對象裝入第1步的list對象中(list.add(map) )。
4.    重複2、3步直到ResultSet遍曆完畢
在DBUtil. resultSetToList(ResultSet rs)方法中實現了上述過程(所有列名均使用大寫),可參考使用。

範例程式碼


  1. //查詢資料部分代碼:
  2.   …
  3.   Connection conn = DBUtil.getConnection();
  4.   PreparedStatement pst = null;
  5.   ResultSet rs = null;
  6.   try{
  7.     String sql=“select emp_code, real_name from t_employee where organ_id=?”;
  8.     pst = conn.preparedStatement(sql);
  9.     pst.setString(1, “101”);
  10.     rs = pst.executeQuery();
  11.     List list = DBUtil. resultSetToList(ResultSet rs);
  12.     return list;
  13.   }finally{
  14.     DBUtil.close(rs, pst ,conn);
  15.   }



  16. //JSP顯示部分代碼
  17. <%
  18.   List empList = (List)request.getAttribute(“empList”);
  19.   if (empList == null) empList = Collections.EMPTY_LIST;
  20. %>
  21. <table  cellspacing="0" width=”90%”>
  22.     <tr>  <td>代碼</td> <td>姓名</td>  </tr>
  23. <%
  24.   Map colMap;
  25.   for (int i=0; i< empList.size(); i++){
  26.     colMap = (Map) empList.get(i);
  27. %>
  28.   <tr>  
  29.     <td><%=colMap.get(“EMP_CODE”)%></td> 
  30.     <td><%=colMap.get(“REAL_NAME”)%></td>  
  31.   </tr>
  32. <%
  33.   }// end for
  34. %>
  35. </table>

解決方案三
    使用RowSet。
RowSet是JDBC2.0 中提供的介面,Oracle對該介面有相應實現,其中很有用的是oracle.jdbc.rowset.OracleCachedRowSet。  OracleCachedRowSet實現了ResultSet中的所有方法,但與ResultSet不同的是,OracleCachedRowSet 中的資料在Connection關閉後仍然有效。

oracle的rowset實現在http://otn.oracle.com/software/content.html的jdbc下載裡有,名稱是ocrs12.zip

範例程式碼

  1. //查詢資料部分代碼:
  2.   import javax.sql.RowSet;
  3.   import oracle.jdbc.rowset.OracleCachedRowSet;
  4.   …
  5.   Connection conn = DBUtil.getConnection();
  6.   PreparedStatement pst = null;
  7.   ResultSet rs = null;
  8.   try{……
  9.     String sql=“select emp_code, real_name from t_employee where organ_id=?”;
  10.     pst = conn.preparedStatement(sql);
  11.     pst.setString(1, “101”);
  12.     rs = pst.executeQuery();
  13.     OracleCachedRowSet ors = newOracleCachedRowSet();
  14.     //將ResultSet中的資料封裝到RowSet中
  15.     ors.populate(rs);
  16.     return ors;
  17.   }finally{
  18.     DBUtil.close(rs, pst, conn);
  19.   }


  20. //JSP顯示部分代碼
  21. <%
  22.   javax.sql.RowSet empRS = (javax.sql.RowSet) request.getAttribute(“empRS”);
  23. %>
  24. <table  cellspacing="0" width=”90%”>
  25.     <tr>  <td>代碼</td> <td>姓名</td>  </tr>
  26. <%
  27.   if (empRS != nullwhile (empRS.next() ) {
  28. %>
  29.   <tr>  
  30.     <td><%= empRS.get(“EMP_CODE”)%></td> 
  31.     <td><%= empRS.get(“REAL_NAME”)%></td>  
  32.   </tr>
  33. <%
  34.   }// end while
  35. %>
  36. </table>

適用場合
  方法一使用於定製的查詢操作
  方法二適用於多條查詢語句或需要對查詢結果進行處理的情況。
  方法三適合於單條查詢語句,適用於快速開發。

相關連結
    如果需要分頁顯示請參考:JSP分頁技術實現
    如果查詢結果需要產生WORD或者EXCEL,請參考:使用jsp實現word、excel格式報表列印

附:DBUtil代碼

  1. import java.util.List;
  2. import java.util.ArrayList;
  3. import java.util.Map;
  4. import java.util.HashMap;
  5. import java.util.Properties;
  6. import java.util.Collections;

  7. import java.sql.Connection;
  8. import java.sql.SQLException;
  9. import java.sql.ResultSet;
  10. import java.sql.ResultSetMetaData;
  11. import java.sql.Statement;
  12. import java.sql.PreparedStatement;

  13. import javax.naming.Context;
  14. import javax.naming.InitialContext;
  15. import javax.naming.NamingException;

  16. import javax.sql.DataSource;

  17. public class DBUtil{

  18.     private static final String JDBC_DATA_SOURCE = "java:comp/env/jdbc/DataSource";

  19.     /**
  20.      enableLocalDebug: 是否在本地調試。<br>
  21.      值為true時如果尋找資料來源失敗則使用DriverManager與資料庫建立串連;
  22.      如果為false則只尋找資料來源建立資料庫連接。
  23.      預設為false。<br>
  24.      可通過系統屬性jdbc.enable_local_debug=true設定enableLocalDebug為true,啟用本地調試:<br>
  25.      增加JVM parameter: -Djdbc.enable_local_debug=true
  26.      */
  27.     private static boolean enableLocalDebug = false;

  28.     static{
  29.         enableLocalDebug = Boolean.getBoolean ("jdbc.enable_local_debug");
  30.     }


  31.     private static Context ctx = null;
  32.     private static javax.sql.DataSource ds = null;


  33.     private static void initDataSource() throws Exception{
  34.         // Put connection properties in to a hashtable.


  35.         if (ctx == null) {
  36.             ctx = new InitialContext();
  37.         }
  38.         if (ds == null) {
  39.             ds = (javax.sql.DataSource) ctx.lookup(JDBC_DATA_SOURCE);
  40.         }
  41.     }        

  42.     /**
  43.      * 尋找應用伺服器資料來源,從資料來源中獲得資料庫連接。<br><br>
  44.      * 在本地調試時如果尋找資料來源失敗並且enableLocalDebug==true
  45.      * 則根據系統屬性使用java.sql.DriverManager建立串連。<br>
  46.      * 本地調試時可配置的系統屬性如下:<br>
  47.      * <p>
  48.      *     #jdbc驅動程式名 <br>
  49.      *     jdbc.driver=<i>oracle.jdbc.driver.OracleDriver</i> <br> <br>
  50.      *     #資料庫連接串<br>
  51.      *     jdbc.url=<i>jdbc:oracle:thin:@10.1.1.1:1521:ocrl</i> <br> <br>
  52.      *     #資料庫使用者名稱<br>
  53.      *     jdbc.username=<i>scott</i> <br> <br>
  54.      *     #資料庫使用者密碼<br>
  55.      *     jdbc.password=<i>tiger</i> <br>
  56.      * </p>
  57.      * 可通過JVM參數設定上述系統屬性:<br>
  58.      * -Djdbc.driver=oracle.jdbc.driver.OracleDriver 
  59.      *  -Djdbc.url=jdbc:oracle:thin:@10.1.1.1:1521:ocrl
  60.      *  -Djdbc.username=scott -Djdbc.password=tiger
  61.      * @return Connection
  62.      * @throws NamingException 如果資料來源尋找失敗
  63.      * @throws SQLException 如果建立資料庫連接失敗
  64.      */
  65.     public static Connection getConnection() throws  SQLException{
  66.         try{
  67.             initDataSource();
  68.             return ds.getConnection();
  69.         }catch(SQLException sqle){
  70.             throw sqle;
  71.         }catch(Exception ne){
  72.             if (enableLocalDebug){
  73.                 return getTestConn();
  74.             }else{
  75.                 throw new RuntimeException(ne.toString());
  76.             }
  77.         }
  78.     }


  79.     //通過DriverManager建立本地測試連接
  80.     private static Connection getTestConn(){
  81.       try {
  82.           String driver = System.getProperty("jdbc.driver");
  83.           System.out.println("jdbc.driver="+driver);

  84.           String url = System.getProperty("jdbc.url");
  85.           System.out.println("jdbc.url="+url);

  86.           String userName = System.getProperty("jdbc.username");
  87.           System.out.println("jdbc.username="+userName);

  88.           String password = System.getProperty("jdbc.password");
  89.           System.out.println("jdbc.password="+password);

  90.           Class.forName(driver).newInstance();
  91.           return java.sql.DriverManager.getConnection(url, userName, password);
  92.       }
  93.       catch (Exception ex) {
  94.         ex.printStackTrace();
  95.         throw new RuntimeException(ex.getMessage());
  96.       }
  97.     }

  98.     /**
  99.      * 將查詢結果封裝成List。<br>
  100.      * List中元素類型為封裝一行資料的Map,Map key為欄位名(大寫),value為相應欄位值
  101.      * @param rs ResultSet
  102.      * @return List
  103.      * @throws java.sql.SQLException
  104.      */
  105.     public static List resultSetToList(ResultSet rs) throws java.sql.SQLException{
  106.         if (rs==nullreturn Collections.EMPTY_LIST;

  107.         ResultSetMetaData md = rs.getMetaData();
  108.         int columnCount = md.getColumnCount();

  109.         List list = new ArrayList();
  110.         Map rowData;
  111.         while (rs.next()){
  112.             rowData = new HashMap(columnCount);
  113.             for (int i=1; i<=columnCount; i++){
  114.                 rowData.put(md.getColumnName(i),rs.getObject(i));
  115.             }
  116.             list.add(rowData);
  117.         }
  118.         return list;
  119.     }

  120.     /**
  121.      * 關閉ResultSet、Statement和Connection
  122.      * @param rs ResultSet to be closed
  123.      * @param stmt Statement or PreparedStatement  to be closed
  124.      * @param conn Connection  to be closed
  125.      */
  126.     public static void close(ResultSet rs, Statement stmt, Connection conn){
  127.             if (rs != nulltry{
  128.                 rs.close();
  129.             }catch(java.sql.SQLException ex){
  130.                 ex.printStackTrace();
  131.             }
  132.             if (stmt != nulltry{
  133.                  stmt.close();
  134.             }catch(java.sql.SQLException ex){
  135.                 ex.printStackTrace();
  136.             }
  137.             if (conn != nulltry{
  138.                 conn.close();
  139.             }catch(java.sql.SQLException ex){
  140.                 ex.printStackTrace();
  141.             }
  142.     }

  143. }// end of DBUti
相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.