分页技术(1)

title:  JSP分页技术实现
summary:使用工具类实现通用分页处理
author:  evan_zhao
email:  evan_zhao@hotmail.com

  目前比较广泛使用的分页方式是将查询结果缓存在HttpSession或有状态bean中,翻页的时候从缓存中取出一页数据显示。这种方法有两个主要的缺点:一是用户可能看到的是过期数据;二是如果数据量非常大时第一次查询遍历结果集会耗费很长时间,并且缓存的数据也会占用大量内存,效率明显下降。
  其它常见的方法还有每次翻页都查询一次数据库,从ResultSet中只取出一页数据(使用rs.last();rs.getRow()获得总计录条数,使用rs.absolute()定位到本页起始记录)。这种方式在某些数据库(如oracle)的JDBC实现中差不多也是需要遍历所有记录,实验证明在记录数很大时速度非常慢。
  至于缓存结果集ResultSet的方法则完全是一种错误的做法。因为ResultSet在Statement或Connection关闭时也会被关闭,如果要使ResultSet有效势必长时间占用数据库连接。

  因此比较好的分页做法应该是每次翻页的时候只从数据库里检索页面大小的块区的数据。这样虽然每次翻页都需要查询数据库,但查询出的记录数很少,网络传输数据量不大,如果使用连接池更可以略过最耗时的建立数据库连接过程。而在数据库端有各种成熟的优化技术用于提高查询速度,比在应用服务器层做缓存有效多了。

  在oracle数据库中查询结果的行号使用伪列ROWNUM表示(从1开始)。例如select  from  employee  where  rownum<10  返回前10条记录。但因为rownum是在查询之后排序之前赋值的,所以查询employee按birthday排序的第100到120条记录应该这么写:
        select from (
            select my_table.*, rownum as my_rownum from (
                select name, birthday from employee order by birthday
            my_table where rownum <120
        where my_rownum>=100

  mySQL可以使用LIMIT子句:
    select  name,  birthday  from  employee  order  by  birthday  LIMIT  99,20
  DB2有rownumber()函数用于获取当前行数。
  SQL  Server没研究过,可以参考这篇文章: http://www.youkuaiyun.com/develop/article/18/18627.shtm

  在Web程序中分页会被频繁使用,但分页的实现细节却是编程过程中比较麻烦的事情。大多分页显示的查询操作都同时需要处理复杂的多重查询条件,sql语句需要动态拼接组成,再加上分页需要的记录定位、总记录条数查询以及查询结果的遍历、封装和显示,程序会变得很复杂并且难以理解。因此需要一些工具类简化分页代码,使程序员专注于业务逻辑部分。下面是我设计的两个工具类:
   PagedStatement    封装了数据库连接、总记录数查询、分页查询、结果数据封装和关闭数据库连接等操作,并使用了PreparedStatement支持动态设置参数。
   RowSetPage    参考PetStore的page  by  page  iterator模式,  设计RowSetPage用于封装查询结果(使用OracleCachedRowSet缓存查询出的一页数据,关于使用CachedRowSet封装数据库查询结果请参考 JSP页面查询显示常用模式)以及当前页码、总记录条数、当前记录数等信息,  并且可以生成简单的HTML分页代码。
  PagedStatement  查询的结果封装成RowsetPage。

  下面是简单的 使用示例
  1.     //DAO查询数据部分代码:
  2.     
  3.     public RowSetPage getEmployee(String gender, int pageNo) throws Exception{
  4.         String sql="select emp_id, emp_code,  user_name, real_name from employee where gender =?";
  5.        //使用Oracle数据库的分页查询实现,每页显示5条
  6.         PagedStatement pst =new PagedStatementOracleImpl(sql,  pageNo, 5);
  7.         pst.setString(1, gender);
  8.         return pst.executeQuery();
  9.     }
  10.     //Servlet处理查询请求部分代码:
  11.     
  12.     int pageNo;
  13.     try{
  14.         //可以通过参数pageno获得用户选择的页码
  15.         pageNo Integer.parseInt(request.getParameter("pageno");
  16.     }catch(Exception ex){
  17.         //默认为第一页
  18.         pageNo=1;
  19.     }
  20.     String gender request.getParameter("gender" );
  21.     request.setAttribute("empPage"myBean.getEmployee(gender, pageNo) );
  22.     
  23.     //JSP显示部分代码
  24. <%@ page import "page.RowSetPage"%>
  25.     
  26.     <script language="javascript">
  27.         function doQuery(){
  28.             form1.actionType.value="doQuery";
  29.             form1.submit();
  30.     }
  31.     </script>
  32.     
  33.     <form name=form1 method=get>
  34.       <input type=hidden name=actionType>
  35.       性别:
  36.       <input type=text name=gender size=1 value="<%=request.getParameter("gender")%>">
  37.       <input type=button value=查询 " onclick="doQuery()">
  38. <%
  39.     RowSetPage empPage (RowSetPage)request.getAttribute("empPage");
  40.     if (empPage == null empPage RowSetPage.EMPTY_PAGE;
  41. %>
  42.     
  43.     <table  cellspacing="0" width="90%">
  44.         <tr> <td>ID</td> <td>代码</td> <td>用户名</td> <td>姓名</td>  </tr>
  45. <%
  46.     javax.sql.RowSet empRS (javax.sql.RowSetempPage.getRowSet();
  47.     if (empRS!=nullwhile (empRS.next() {
  48. %>
  49.         <tr>  
  50.             <td><%= empRS.getString("EMP_ID")%></td> 
  51.             <td><%= empRS.getString("EMP_CODE")%></td>  
  52.             <td><%= empRS.getString("USER_NAME")%></td> 
  53.             <td><%= empRS.getString("REAL_NAME")%></td>  
  54.         </tr>
  55. <%
  56.     }// end while
  57. %>
  58.         <tr>
  59. <%
  60.     //显示总页数和当前页数(pageno)以及分页代码。
  61.     //此处doQuery为页面上提交查询动作的javascript函数名, pageno为标识当前页码的参数名
  62. %>
  63.             <td colspan=4><%= empPage .getHTML("doQuery""pageno")%></td>
  64.         </tr>
  65.     </table>
  66.     </form>

  效果如图:
分页技术(1)

  因为分页显示一般都会伴有查询条件和查询动作,页面应已经有校验查询条件和提交查询的javascript方法(如上面的doQuery),所以RowSetPage.getHTML()生成的分页代码在用户选择新页码时直接回调前面的处理提交查询的javascript方法。注意在显示查询结果的时候上次的查询条件也需要保持,如<input  type=text  name=gender  size=1  value="<%=request.getParameter("gender")%>">。同时由于页码的参数名可以指定,因此也支持在同一页面中有多个分页区。
  另一种分页代码实现是生成每一页的URL,将查询参数和页码作为QueryString附在URL后面。这种方法的缺陷是在查询条件比较复杂时难以处理,并且需要指定处理查询动作的servlet,可能不适合某些定制的查询操作。
  如果对RowSetPage.getHTML()生成的默认分页代码不满意可以编写自己的分页处理代码,RowSetPage提供了很多getter方法用于获取相关信息(如当前页码、总页数、  总记录数和当前记录数等)。
  在实际应用中可以将分页查询和显示做成jsp  taglib,  进一步简化JSP代码,屏蔽Java  Code。

附:分页工具类的源代码,  有注释,应该很容易理解。

1.Page.java
2.RowSetPage.java(RowSetPage继承Page)
3.PagedStatement.java
4.PagedStatementOracleImpl .java(PagedStatementOracleImpl 继承PagedStatement)



您可以任意使用这些源代码,但必须保留author  evan_zhao@hotmail.com字样
  1. ///////////////////////////////////
  2. //
  3. //  Page.java
  4. //  author: evan_zhao@hotmail.com
  5. //
  6. ///////////////////////////////////
  7. package page;
  8. import java.util.List;
  9. import java.util.ArrayList;
  10. import java.util.Collection;
  11. import java.util.Collections;
  12. public  class Page implements java.io.Serializable {
  13.     public static final Page EMPTY_PAGE new Page();
  14.     public static final int  DEFAULT_PAGE_SIZE 20;
  15.     public static final  int MAX_PAGE_SIZE 9999;
  16.     private int myPageSize DEFAULT_PAGE_SIZE;
  17.     private int start;
  18.     private int avaCount,totalSize;
  19.     private Object data;
  20.     private int currentPageno;
  21.     private int totalPageCount;
  22.     
  23.     protected Page(){
  24.         this.init(0,0,0,DEFAULT_PAGE_SIZE,new Object());
  25.     }
  26.     
  27.     protected void init(int start, int avaCount, int totalSize, int pageSize, Object data){
  28.         this.avaCount =avaCount;
  29.         this.myPageSize pageSize;
  30.         this.start start;
  31.         this.totalSize totalSize;
  32.         this.data=data;
  33.         //System.out.println("avaCount:"+avaCount);
  34.         //System.out.println("totalSize:"+totalSize);
  35.         if (avaCount>totalSize) {
  36.             //throw new RuntimeException("记录条数大于总条数?!");
  37.         }
  38.         this.currentPageno (start -1)/pageSize +1;
  39.         this.totalPageCount (totalSize pageSize -1) pageSize;
  40.         if (totalSize==0 && avaCount==0){
  41.             this.currentPageno 1;
  42.             this.totalPageCount 1;
  43.         }
  44.         //System.out.println("Start Index to Page No: start "-" currentPageno);
  45.     }
  46.     public  Object getData(){
  47.         return this.data;
  48.     }
  49.     
  50.     public int getPageSize(){
  51.         return this.myPageSize;
  52.     }
  53.     
  54.     public boolean hasNextPage() {
  55.       
  56.       return (this.getCurrentPageNo()<this.getTotalPageCount());
  57.     }
  58.     
  59.     public boolean hasPreviousPage() {
  60.       
  61.       return (this.getCurrentPageNo()>1);
  62.     }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值