网页分页功能的实现

时间:2022-09-20 20:41:03

最近在学习JavaWeb的时候,用到了分页功能,现在进行一个记录,以备不时之需

第一步:先完成PageBean的编写

就是对当前页数,每页显示的记录数,总记录数,总页数,分页显示的信息进行封装。作为通用的分页功能的实现,这里用到了泛型

import java.util.List;

/**
* 分页封装
*
*/
public class PageBean<T> {

private int currPage;//当前页数

private int pageSize;//每页显示记录数

private int totalCount;//总记录数

private int totalPage;//总页数

private List<T> list;//每页显示的信息

public int getCurrPage() {
return currPage;
}

public void setCurrPage(int currPage) {
this.currPage = currPage;
}

public int getPageSize() {
return pageSize;
}

public void setPageSize(int pageSize) {
this.pageSize = pageSize;
}

public int getTotalCount() {
return totalCount;
}

public void setTotalCount(int totalCount) {
this.totalCount = totalCount;
}

public int getTotalPage() {
return totalPage;
}

public void setTotalPage(int totalPage) {
this.totalPage = totalPage;
}

public List<T> getList() {
return list;
}

public void setList(List<T> list) {
this.list = list;
}

}

 

第二步:在Action类中编写一个分页方法

Action中通过调用业务层Service类的分页方法,Employee就是具体的信息Bean,之前泛型的使用就在于此,在不同项目中使用不同的信息Bean可以完成多种类信息的分页,employeeService就是具体的业务层Service,Service中我们也有一个叫findAll的方法。这里的currPage生成其set/get方法,等用户点击了页面及具体的页数,Struts2会获得具体的页数,然后将currPage传给Service

    // 分页(当前页)这里等于1是为了使第一页为默认页
private int currPage = 1;
public int getCurrPage() {
return currPage;
}

public void setCurrPage(int currPage) {
this.currPage = currPage;
}
/**
* 分页查询
*/
public String findAll() {
PageBean
<Employee> pageBean = employeeService.findAll(currPage);
ActionContext.getContext().getValueStack().push(pageBean);
return "findAll";
}

 

第三步:编写Service中的分页方法

先生成一个PageBean对象(注意泛型),之后开始封装这个Bean。pageSize(每页显示的记录数)这里设置为3,就是每页显示3条记录,totalCount(总记录数)通过Dao中的findCount()方法来查到,totalPage(总页数)=totalCount(总记录数)/pageSize(每页大小),Math.ceil可以获得一个double的近似值(大于等于),之后我们通过Double包装类的intValue再转成int型,begin(每一页的开头的序号),list的数据也通过Dao的findPage(int,int)方法获得

public PageBean<Employee> findAll(int currPage) {
PageBean
<Employee> pageBean = new PageBean<>();
// 封装pageBean
pageBean.setCurrPage(currPage);
int pageSize = 3;
pageBean.setPageSize(pageSize);
int totalCount = employeeDao.findCount();
pageBean.setTotalCount(totalCount);
Double totalPage
= Math.ceil((double) totalCount / pageSize);
pageBean.setTotalPage(totalPage.intValue());
int begin = (currPage - 1) * pageSize;
List
<Employee> list = employeeDao.findPage(begin, pageSize);
pageBean.setList(list);
return pageBean;
}

 

第四步:编写Dao中的方法

这之前在Service中使用的findCount()和findPage(int,int),值得注意的就是findCount()中的hql语句使用count(*),还有就是findPage(int,int)中使用了org.hibernate.criterion.DetachedCriteria类,查询时不是用find方法,而是findByCriteria来查询

   /**
* 查询总记录数
*/
public int findCount() {
String hql
="select count(*) from Employee";
List
<Long> list = (List<Long>) hibernateTemplate.find(hql);
if(list.size()>0){
return list.get(0).intValue();
}
return 0;
}


/**
* 分页信息
*/
public List<Employee> findPage(int begin, int pageSize) {
DetachedCriteria criteria
=DetachedCriteria.forClass(Employee.class);
List
<Employee> list = (List<Employee>) hibernateTemplate.findByCriteria(criteria, begin, pageSize);
return list;
}

基本上通过以上四步就完成了分页功能逻辑代码的编写,接下来就是对视图层页面的编写

下面是个分页的简单显示,处于首页时,只显示下一页和尾页,处于尾页时,只显示首页和上一页,其他页就都显示。这里使用的Struts2的标签库,所以不要忘了要加上

<%@ taglib uri="/struts-tags" prefix="s" %>

为什么我们可以直接currPage,totalPage等属性?是因为我们之前将PageBean对象放入了值栈中

<table border="0" cellspacing="0" cellpadding="0"  width="900px">
<tr>
<td align="right">
<span><s:property value="currPage"/>/<s:property value="totalPage"/></span>&nbsp;&nbsp;
<span>总记录数:<s:property value="totalCount"/>/每页显示:<s:property value="pageSize"/></span>&nbsp;&nbsp;
<span>
<s:if test="currPage!=1">
<a href="department_findAll.action?currPage=1">[首页]</a>&nbsp;&nbsp;
<a href="department_findAll.action?currPage=<s:property value="currPage-1"/>">[上一页]</a>&nbsp;&nbsp;
</s:if>
<s:if test="currPage!=totalPage">
<a href="department_findAll.action?currPage=<s:property value="currPage+1"/>">[下一页]</a>&nbsp;&nbsp;
<a href="department_findAll.action?currPage=<s:property value="totalPage"/>">[尾页]</a>&nbsp;&nbsp;
</s:if>
</span>
</td>
</tr>
</table>

效果图:

网页分页功能的实现

可以使用了struts2的<s:iterator>标签来进行迭代显示数据,这里给个参考

<s:iterator value="list" var="d">
<tr>
<td align="center"><s:property value="#d.dname"/></td>
<td align="center"><a href="">编辑</a></td>
<td align="center"><a href="">删除</a></td>
</tr>
</s:iterator>

 注意:一定要在Struts.xml文件中配置转跳到我们编写的action上(在这里是department_findAll.action),不然打开要分页的页面时,不会进行分页操作,只有你选择了页数才会分页

网页分页功能的实现

 

后记

我在使用的过程中发现,要使用分页功能的地方不少,在某些地方,使用上面的会出错:例如对条件查询之后的结果进行分页,第一页很OK,但是你点击下一页/某一页的时候会出现查询条件的丢失问题,再进行的查询结果分页是没有条件的,就会出错

解决办法(我使用Servlet+JDBC实现的,原理是一样的):

在我们的POJO中添加一个url属性,表示查询条件,因为POST方式条件是放在请求头中的,很不方便,所以再进行条件查询的的表单使用GET方式:

<form action="<c:url value='/customerServlet'/>" method="get">

我们需要获得这个url(包括项目名+Servlet名+条件),获得这个url封装到POJO中

改进后的PageBean:

import java.util.List;

public class PageBean<T> {
//当前页
private int currPage;
//每页记录数
private int pageSize;
//总记录数
private int totalCount;
//数据集合
private List<T> list;
//url表示条件查询的条件(GET方式)
private String url;

public int getCurrPage() {
return currPage;
}

public void setCurrPage(int currPage) {
this.currPage = currPage;
}

public int getPageSize() {
return pageSize;
}

public void setPageSize(int pageSize) {
this.pageSize = pageSize;
}

public int getTotalCount() {
return totalCount;
}

public void setTotalCount(int totalCount) {
this.totalCount = totalCount;
}

//设置总页数(计算得出)
public int getTotalPage() {
Double totalPage
=Math.ceil((double)totalCount/pageSize);
return totalPage.intValue();
}

public List<T> getList() {
return list;
}

public void setList(List<T> list) {
this.list = list;
}

public String getUrl() {
return url;
}

public void setUrl(String url) {
this.url = url;
}
}

控制层要添加的方法(encoding()方法不一定需要):

    /**
* 得到包含查询的条件URL
*/
private String getURL(HttpServletRequest request){
String contextPath
=request.getContextPath(); //项目名
String servletPath=request.getServletPath(); //servlet路径,即/*Servlet 建议使用request.getRequestURI()获得一个包含项目名+Servlet的路径
String queryString=request.getQueryString(); //?后面的参数
//判断参数中是否带当前页(currPage)
if(queryString.contains("&currPage=")){
int index=queryString.lastIndexOf("&currPage=");
queryString
=queryString.substring(0, index);
}
return contextPath+servletPath+"?"+queryString;
}

/**
*对(这里是4个条件)条件进行编码(GET方式防止中文乱码,POST方式已经在BaseServlet中配置)
*
@throws UnsupportedEncodingException
*
*/
private Customer encoding(Customer customer) throws UnsupportedEncodingException{
String cname
=customer.getCname();
String gender
=customer.getGender();
String cellPhone
=customer.getCellphone();
String email
=customer.getEmail();
if(cname!=null||!cname.isEmpty()){
cname
=new String(cname.getBytes("ISO-8859-1"),"UTF-8");
customer.setCname(cname);
}
if(gender!=null||!gender.isEmpty()){
gender
=new String(gender.getBytes("ISO-8859-1"),"UTF-8");
customer.setCname(gender);
}
if(cellPhone!=null||!cellPhone.isEmpty()){
cellPhone
=new String(cellPhone.getBytes("ISO-8859-1"),"UTF-8");
customer.setCname(cellPhone);
}
if(email!=null||!email.isEmpty()){
email
=new String(email.getBytes("ISO-8859-1"),"UTF-8");
customer.setCname(email);
}
return customer;
}

这个geturl()方法就是获得url的方法,我用来表示当前页的变量使用的是currPage,根据实际情况更换。encoding()方法是处理get方式获得的条件中文乱码问题,如果项目使用的就是UTF-8就不用转码了,我写的encoding()方法是从ISO-8859-1转成UTF-8,其他的格式转码类似。注意:现在是所有的查询的方法的需要使用geturl()方法,并将返回值封装到POJO中。使用request.getRequestURI()方法也可以获得Struts2中的Action。使用框架开发应该就不用考虑encoding()这个编码方法,只需要处理geturl()方法了

多条件查询的Servlet方法:

    /**
* 多条件查询
*/
public String query(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//获得查询条件
Customer customer=CommonUtils.toBean(request.getParameterMap(), Customer.class);
//处理GET请求的编码格式
//customer = encoding(customer); 因为我的项目使用就是UTF-8,所以就不转码了
//处理PageBean
int pageSize=10; //每页记录数为10
int currPage=getCurrentPage(request);
int begin=(currPage-1)*pageSize; //计算每页的开头
PageBean<Customer> pageBean=customerService.query(customer,begin,pageSize);
//封装当前页和每页记录数
pageBean.setCurrPage(currPage);
pageBean.setPageSize(pageSize);
//封装url
pageBean.setUrl(getURL(request));
//将pageBean保存request
request.setAttribute("pb", pageBean);
return "f:/list.jsp";
}

 

Dao中多条件查询:

/**
* 多条件查询
*
*/
public PageBean<Customer> query(Customer customer, int begin, int pageSize) {
try {
PageBean
<Customer> pageBean=new PageBean<Customer>();

/**
* 根据条件查询出总记录数
* 拼sql语句
*/
StringBuilder cntSql
=new StringBuilder("select count(*) from t_customer");
StringBuilder whereSql
=new StringBuilder(" where 1=1");
List
<Object> params=new ArrayList<Object>();

String cname
=customer.getCname();
if(cname!=null&&!cname.trim().isEmpty()){
whereSql.append(
" and cname like ?");
params.add(
"%"+cname+"%");
}

String gender
=customer.getGender();if(gender!=null&&!gender.trim().isEmpty()){
whereSql.append(
" and gender=?");
params.add(gender);
}

String phone
=customer.getCellphone();
if(phone!=null&&!phone.trim().isEmpty()){
whereSql.append(
" and cellphone like ?");
params.add(
"%"+phone+"%");
}

String email
=customer.getEmail();
if(email!=null&&!email.trim().isEmpty()){
whereSql.append(
" and email like ?");
params.add(
"%"+email+"%");
}
//运行sql
Number count=(Number) qr.query(cntSql.append(whereSql).toString(), new ScalarHandler(),params.toArray());
pageBean.setTotalCount(count.intValue());


/**
* 根据条件查询出每页数据
*/
StringBuilder listSql
=new StringBuilder("select * from t_customer");
//limit子句分页
StringBuilder limitSql=new StringBuilder(" order by cname limit ?,?");
//添加这两个参数
params.add(begin);
params.add(pageSize);
//运行sql
List<Customer> beanList=qr.query(listSql.append(whereSql).append(limitSql).toString(),
new BeanListHandler<Customer>(Customer.class),
params.toArray());
pageBean.setList(beanList);
return pageBean;
}
catch (Exception e) {
e.printStackTrace();
}
return null;
}

我还是使用的是拼接sql的方式,与众不同的是分页中使用的count语句limit语句都是带where条件的

 

 在页面中也需要进行相应的修改(我这里使用的是JSTL):

第${pb.currPage }页/共${pb.totalPage }页
<c:if test="${pb.currPage>1 }">
<a href="${pb.url }&currPage=1">首页</a>
<a href="${pb.url }&currPage=${pb.currPage-1 }">上一页</a>
</c:if>

<c:if test="${pb.currPage<pb.totalPage }">
<a href="${pb.url }&currPage=${pb.currPage+1 }">下一页</a>
<a href="${pb.url }&currPage=${pb.totalPage }">尾页</a>
</c:if>

pb就是我放在request中的pageBean的名字,注意url需要我们从pageBean的url中取出,页面只关心当前页,不去关心路径到底是什么

 

一个页码的实现

可能说一个页码又什么好实现的,直接把所有页数写出来,都加上超链接不就好了,但是观察百度等有页码的网站,你会发现页面是会随着页数变换的,比如说:先显示10页,但是当你点击第7页的时候,第1页应该隐藏,第11页应该出现,只有出现11页我们才能点击,这就是要实现的功能

<%-- 页码 --%>
<c:choose>
<%-- 总页数小于等于10页,把所有页显示 --%>
<c:when test="${pb.totalPage<=10 }">
<c:set var="begin" value="1"/>
<c:set var="end" value="${pb.totalPage }"/>
</c:when>
<c:otherwise>
<%--总页数>10,计算begin和end --%>
<c:set var="begin" value="${pb.currPage-5 }"/>
<c:set var="end" value="${pb.currPage+4 }"/>
<%--处理头溢出 --%>
<c:if test="${begin<1 }">
<c:set var="begin" value="1"/>
<c:set var="end" value="10"/>
</c:if>
<%-- 处理尾溢出 --%>
<c:if test="${end>pb.totalPage }">
<c:set var="begin" value="${pb.totalPage-9 }"/>
<c:set var="end" value="${pb.totalPage }"/>
</c:if>
</c:otherwise>
</c:choose>
<c:forEach begin="${begin }" end="${end }" var="i">
<c:choose>
<c:when test="${i eq pb.currPage }">
[${i }]
</c:when>
<c:otherwise>
<a href="${pb.url }&currPage=${i }">[${i }]</a>
</c:otherwise>
</c:choose>
</c:forEach>

这是一个显示10页,点击第7页后,1页消失11页出现,点击8页后,2页消失12页出现,以此类推

效果图(这是我把页码放在上一页和下一页之间的效果):

网页分页功能的实现