JSP分页技术实现

转载 2006年06月07日 22:09:00
目前比较广泛使用的分页方式是将查询结果缓存在HttpSession或有状态bean中,翻页的时候从缓存中取出一页数据显示。这种方法有两个主要的缺点:一是用户可能看到的是过期数据;二是如果数据量非常大时第一次查询遍历结果集会耗费很长时间,并且缓存的数据也会占用大量内存,效率明显下降。
  其它常见的方法还有每次翻页都查询一次数据库,从ResultSet中只取出一页数据(使用rs.last();rs.getRow()获得总计录条数,使用rs.absolute()定位到本页起始记录)。这种方式差不多也是需要遍历所有记录,实验证明在记录数很大时速度非常慢。
  至于缓存结果集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.csdn.net/develop/article/18/18627.shtm

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

  下面是简单的使用示例



public RowSetPage getEmployee(String gender, int pageNo) throws Exception{
String sql=;

PagedStatement pst =new PagedStatementOracleImpl(sql, pageNo, 5);
pst.setString(1, gender);
return pst.executeQuery();
}





int pageNo;
try{

pageNo = Integer.parseInt(request.getParameter() );
}catch(Exception ex){

pageNo=1;
}
String gender = request.getParameter( );
request.setAttribute(, myBean.getEmployee(gender, pageNo) );



<%@ page import = %>

<script language=>
function doQuery(){
form1.actionType.value=;
form1.submit();
}
</script>

<form name=form1 method=get>
<input type=hidden name=actionType>
性别:
<input type=text name=gender size=1 value=gender>
<input type=button value= onclick=>
</form>
<%
RowSetPage empPage = (RowSetPage)request.getAttribute();
if (empPage == null ) empPage = RowSetPage.EMPTY_PAGE;
%>

<table cellspacing= width=>
<tr> <td>ID</td> <td>代码</td> <td>用户名</td> <td>姓名</td> </tr>
<%
javax.sql.RowSet empRS = (javax.sql.RowSet) empPage.getRowSet();
if (empRS!=null) while (empRS.next() ) {
%>
<tr>
<td><%= empRS.getString()%></td>
<td><%= empRS.getString()%></td>
<td><%= empRS.getString()%></td>
<td><%= empRS.getString()%></td>
</tr>
<%
}
%>
<tr>
<%


%>
<td colspan=4><%= empPage .getHTML(, )%></td>
</tr>
</table>

  效果如图:


  因为分页显示一般都会伴有查询条件和查询动作,页面应已经有校验查询条件和提交查询的java script方法(如上面的doQuery),所以RowSetPage.getHTML()生成的分页代码在用户选择新页码时直接回调前面的处理提交查询的java script方法。注意在显示查询结果的时候上次的查询条件也需要保持,如<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字样








package page;

import java.util.List;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;



public class Page implements java.io.Serializable {
public static final Page EMPTY_PAGE = new Page();
public static final int DEFAULT_PAGE_SIZE = 20;
public static final int MAX_PAGE_SIZE = 9999;

private int myPageSize = DEFAULT_PAGE_SIZE;

private int start;
private int avaCount,totalSize;
private Object data;

private int currentPageno;
private int totalPageCount;


protected Page(){
this.init(0,0,0,DEFAULT_PAGE_SIZE,new Object());
}


protected void init(int start, int avaCount, int totalSize, int pageSize, Object data){

this.avaCount =avaCount;
this.myPageSize = pageSize;

this.start = start;
this.totalSize = totalSize;

this.data=data;



if (avaCount>totalSize) {

}

this.currentPageno = (start -1)/pageSize +1;
this.totalPageCount = (totalSize + pageSize -1) / pageSize;

if (totalSize==0 && avaCount==0){
this.currentPageno = 1;
this.totalPageCount = 1;
}

}

public Object getData(){
return this.data;
}


public int getPageSize(){
return this.myPageSize;
}


public boolean hasNextPage() {

return (this.getCurrentPageNo()<this.getTotalPageCount());
}


public boolean hasPreviousPage() {

return (this.getCurrentPageNo()>1);
}


public int getStart(){
return start;
}


public int getEnd(){
int end = this.getStart() + this.getSize() -1;
if (end<0) {
end = 0;
}
return end;
}


public int getStartOfPreviousPage() {
return Math.max(start-myPageSize, 1);
}



public int getStartOfNextPage() {
return start + avaCount;
}


public static int getStartOfAnyPage(int pageNo){
return getStartOfAnyPage(pageNo, DEFAULT_PAGE_SIZE);
}


public static int getStartOfAnyPage(int pageNo, int pageSize){
int startIndex = (pageNo-1) * pageSize + 1;
if ( startIndex < 1) startIndex = 1;

return startIndex;
}


public int getSize() {
return avaCount;
}


public int getTotalSize() {
return this.totalSize;
}


public int getCurrentPageNo(){
return this.currentPageno;
}


public int getTotalPageCount(){
return this.totalPageCount;
}



public String getHTML(String queryJSFunctionName, String pageNoParamName){
if (getTotalPageCount()<1){
return +pageNoParamName+;
}
if (queryJSFunctionName == null || queryJSFunctionName.trim().length()<1) {
queryJSFunctionName = ;
}
if (pageNoParamName == null || pageNoParamName.trim().length()<1){
pageNoParamName = ;
}

String gotoPage = +queryJSFunctionName;

StringBuffer html = new StringBuffer();
html.append()
.append().append(gotoPage).append()
.append( )
.append( )
.append(pageNoParamName).append()
.append( ).append(pageNoParamName)
.append()
.append( ).append(queryJSFunctionName).append()
.append( )
.append( )



.append( )
.append(queryJSFunctionName).append()
.append( ).append(pageNoParamName)
.append()
.append( )

.append( )
.append( )
.append( )
.append( );
html.append( )
.append( )
.append( );
html.append( ).append( getTotalPageCount() ).append( )
.append( ) .append(getStart()).append().append(getEnd())
.append().append(this.getTotalSize()).append()
.append( )
.append( );
if (hasPreviousPage()){
html.append( ).append(gotoPage)
.append() .append(getCurrentPageNo()-1)
.append( );
}
html.append( )
.append( )
.append(pageNoParamName).append()
.append(gotoPage).append();
String selected = ;
for(int i=1;i<=getTotalPageCount();i++){
if( i == getCurrentPageNo() )
selected = ;
else selected = ;
html.append( ).append(i).append()
.append(selected).append().append(i).append();
}
if (getCurrentPageNo()>getTotalPageCount()){
html.append( ).append(getCurrentPageNo())
.append().append(getCurrentPageNo())
.append();
}
html.append( );
if (hasNextPage()){
html.append( ).append(gotoPage)
.append().append((getCurrentPageNo()+1))
.append( );
}
html.append( );

return html.toString();

}
}

JSP分页技术实现

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

使用JSP开发技术实现分页效果

  • 2011年06月11日 23:28
  • 1KB
  • 下载

JSP页面分页技术实现

  • 2013年12月06日 22:06
  • 333KB
  • 下载

JSP学习之------>JSP分页技术实现 [转贴]

目前比较广泛使用的分页方式是将查询结果缓存在HttpSession或有状态bean中,翻页的时候从缓存中取出一页数据显示。这种方法有两个主要的缺点:一是用户可能看到的是过期数据;二是如果数据量非常大时...

ajax技术实现分页

  • 2011年07月05日 18:41
  • 113KB
  • 下载

分页技术实现与分析

  • 2013年08月09日 14:30
  • 1.03MB
  • 下载

easyui 中的datagrid分页技术实现

一些基本的知识参考官网          http://www.jeasyui.com/documentation/index.php# 这里我强调一点 datagrid会...

仿谷歌,百度查询页面技术实现分页分析

分页 (pagination) 一种自动分页机制,可以将移动 Web 窗体中的内容分割成一组组较小的页进行呈现,以适合于特定的设备。该机制还呈现可用于浏览到其他页的用户界面元素.在整个的web开发应用...

仿Baidu,Google查询分页技术实现分析

分页 (pagination) 一种自动分页机制,可以将移动 Web 窗体中的内容分割成一组组较小的页进行呈现,以适合于特定的设备。该机制还呈现可用于浏览到其他页的用户界面元素.在整个的web开发应用...
  • mm2223
  • mm2223
  • 2011年11月03日 10:35
  • 314

【mfc】利用单一对话框内的分页技术实现向导功能

所谓的向导功能,在win32的程序中常常见于安装程序或者程序内的设置向导,该向导能够指引用户去完成一些事情,不用用户一次性面对一大堆复杂的设置内容。同时,最主要的功能是现实单一对话框内的分页技术,如果...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:JSP分页技术实现
举报原因:
原因补充:

(最多只允许输入30个字)