首先讓我們構(gòu)建一個(gè)PageControl對(duì)象將分頁(yè)所涉及到的一些關(guān)鍵的"控制數(shù)據(jù)"予以封裝.
具體說(shuō)明如下:
public int curPage ; //當(dāng)前是第幾頁(yè)
public int maxPage ; //一共有多少頁(yè)
public int maxRowCount ; //一共有多少行
public int rowsPerPage ; //每頁(yè)有多少行
public yourDataType yourdata ;//裝載每頁(yè)的數(shù)據(jù)
關(guān)于每頁(yè)所要顯示的"實(shí)體數(shù)據(jù)"的載體,其實(shí)現(xiàn)方式多種多樣,比如說(shuō)在IBM電子商務(wù)系統(tǒng)MPE中是以bean的形式,這是一種面向?qū)ο蟮膶?shí)現(xiàn),比較的簡(jiǎn)略的實(shí)現(xiàn)可用java.util.Vector等,為了避免分散對(duì)核心問(wèn)題的的注意力,這里用youDataType予以抽象.
public void countMaxPage() { //根據(jù)總行數(shù)計(jì)算總頁(yè)數(shù)
if (this.maxRowCount % this.rowsPerPage==0){
this.maxPage = this.maxRowCount/this.rowsPerPage;
}else{
this.maxPage = this.maxRowCount/this.rowsPerPage + 1;
}
}
this.rowsPerPage其實(shí)應(yīng)從配置文件中獲得,這樣做的好處是程序能在運(yùn)行中讀取從而實(shí)現(xiàn)動(dòng)態(tài)(再)配置,簡(jiǎn)略的做法是直接寫(xiě)在程序中。
public PageControl(yourPersistenceLayer yourPL)
這是一個(gè)參數(shù)類型為yourPersistenceLayer的構(gòu)造函數(shù).PersistenceLayer是直接同數(shù)據(jù)庫(kù)打交道的一層,不同的公司都有不同的實(shí)現(xiàn),比如說(shuō)Microsoft的ADO就可以看作是一PersistenceLayer,IBM在其MPE系統(tǒng)中也實(shí)現(xiàn)了一個(gè)龐大的PersistenceLayer,. 一種投機(jī)的做法是不要PersistenceLayer,或者可以說(shuō)是淡化該層,這樣做勢(shì)必降低系統(tǒng)的穩(wěn)定性,可重用性,可擴(kuò)展性。 具體可以參考附錄文獻(xiàn).在這個(gè)構(gòu)造函數(shù)中,有這樣幾個(gè)主要操作:
this.maxRowCount = yourPL.getAvailableCount(); //得到總行數(shù)
this.yourdata = yourPL.getResult(); //得到要顯示于本頁(yè)的數(shù)據(jù)
this.countMaxPage(); //計(jì)算總頁(yè)數(shù)
關(guān)于this.yourdata這里還有一個(gè)細(xì)節(jié):在從數(shù)據(jù)庫(kù)中獲取"實(shí)體數(shù)據(jù)"時(shí),通常有兩種方式:(A)一次性獲取所有數(shù)據(jù);(B)每次根據(jù)當(dāng)前頁(yè)號(hào),獲取本頁(yè)的數(shù)據(jù),將其它數(shù)據(jù)予以拋棄;考慮到數(shù)據(jù)往往是大量甚至是海量的,如果一次性的獲取,那么這些數(shù)據(jù)必然大量占用服務(wù)器內(nèi)存資源,使系統(tǒng)性能大大降低,因此建議使用方法(A)
接下來(lái)的工作就可以交給servlet和jsp了
在servlet的service()方法中只需進(jìn)行如下操作:
PageControl pageCtl = yourBusinessObject.listData(req.getParameter("jumpPage"));
req.setAttribute("pageCtl",pageCtl);
說(shuō)明:yourBusinessObject封裝了商業(yè)邏輯,是位于Business Logic Layer中的一個(gè)對(duì)象,運(yùn)用OOAD的方法,封裝商業(yè)對(duì)象,在Persistent Layer之上組建堅(jiān)實(shí)的Business Logic Layer同樣是構(gòu)建大型電子商務(wù)架構(gòu)的關(guān)鍵所在。本文的關(guān)注點(diǎn)只是分頁(yè)處理,暫不詳細(xì)論述.
在每個(gè)想要實(shí)現(xiàn)翻頁(yè)顯示數(shù)據(jù)的jsp頁(yè)面中,我們的工作也很簡(jiǎn)單,其代碼是公式化的:
<jsp:useBean id="pageCtl" class="yourpackage.PageControl" scope="request"></jsp:useBean>
<%if(pageCtl.maxPage!=1)){%>
<form name="PageForm" action="/servlet/yourpackage.yourservlet" method="post">
<%@ include file="/yourpath/pageman.jsp"%>
</form>
<%}%>
說(shuō)明:
if(pageCtl.maxPage!=1)實(shí)現(xiàn)了這樣一個(gè)邏輯:如果所取得數(shù)據(jù)不足一頁(yè),那么就不用進(jìn)行翻頁(yè)顯示。
我們注意到<%@ include file="/yourpath/pageman.jsp"%>這使得真正的翻頁(yè)部分完全得到了重用.
那么pageman.jsp到底做了些什么呢?它實(shí)現(xiàn)了經(jīng)常做翻頁(yè)處理的人耳熟能詳?shù)倪壿?nbsp;
(A)第一頁(yè)時(shí)不能再向前翻;
(B)最后一頁(yè)時(shí)不能再向后翻;
同時(shí)能夠進(jìn)行頁(yè)面任意跳轉(zhuǎn),具體代碼如下:
每頁(yè)<%=pageCtl.rowsPerPage%>行
共<%=pageCtl.maxRowCount%>行
第<%=pageCtl.curPage%>頁(yè)
共<%=pageCtl.maxPage%>頁(yè)
<%if(pageCtl.curPage==1){ out.print(" 首頁(yè) 上一頁(yè)"); }else{ %>
<A HREF="javascript:gotoPage(1)">首頁(yè)
<A HREF="javascript:gotoPage(<%=pageCtl.curPage-1%>)">上一頁(yè)
<%}%>
<%if(pageCtl.curPage==pageCtl.maxPage){ out.print("下一頁(yè) 尾頁(yè)"); }else{ %>
<A HREF="javascript:gotoPage(<%=pageCtl.curPage+1%>)">下一頁(yè)
<A HREF="javascript:gotoPage(<%=pageCtl.maxPage%>)">尾頁(yè)
<%}%>
轉(zhuǎn)到第<SELECT name="jumpPage" onchange="Jumping()">
<% for(int i=1;i<=pageCtl.maxPage;i++) {
if (i== pageCtl.curPage){
%>
<OPTION selected value=<%=i%>><%=i%></OPTION>
<%}else{%>
<OPTION value=<%=i%>><%=i%></OPTION>
<%}}%>
</SELECT>頁(yè)
在頁(yè)面上會(huì)呈現(xiàn)出形如下圖的外觀,至于想要修飾美化,那是美工的工作所在。
最后附上用于頁(yè)面跳轉(zhuǎn)的javascript公共函數(shù):
function Jumping(){
document.PageForm.submit();
return ;
}
function gotoPage(pagenum){
document.PageForm.jumpPage.value = pagenum;
document.PageForm.submit();
return ;
}
如果他們位于名為T(mén)urnPage.js的文件中,那么在每個(gè)要實(shí)現(xiàn)翻頁(yè)數(shù)據(jù)顯示的jsp頁(yè)面中加上這樣的引用:
<SCRIPT Language="JavaScript" src="/yourPath/TurnPage.js"></SCRIPT>