前言 PA-0FlV|
在使用数据库的过程中,不可避免的需要使用到分页的功能,可是JDBC的规范对此却没有很好的解决。对于这个需求很多朋友都有自己的解决方案,比如使用Vector等集合类先保存取出的数据再分页。但这种方法的可用性很差,与JDBC本身的接口完全不同,对不同类型的字段的支持也不好。这里提供了一种与JDBC兼容性非常好的方案。 {ZIFj.2
JDBC和分页 HOQ
_T4
Sun的JDBC规范的制定,有时很让人哭笑不得,在JDBC1.0中,对于一个结果集(ResultSet)你甚至只能执行next()操作,而无法让其向后滚动,这就直接导致在只执行一次SQL查询的情况下无法获得结果集的大小。所以,如果你使用的是JDBC1.0的驱动,那么是几乎无法实现分页的。 :~A1Ud4c
好在Sun的JDBC2规范中很好的弥补了这一个不足,增加了结果集的前后滚动操作,虽然仍然不能直接支持分页,但我们已经可以在这个基础上写出自己的可支持分页的ResultSet了。 hr}R,BR|
Ef*.}gcU
3L!&~'.Ro
nTtt$I@hW
和具体数据库相关的实现方法 yNMwd.r[
有一些数据库,如Mysql, Oracle等有自己的分页方法,比如Mysql可以使用limit子句,Oracle可以使用ROWNUM来限制结果集的大小和起始位置。这里以Mysql为例,其典型代码如下: I3[RaZ2z{
// 计算总的记录条数 OFAqP1o{$
String SQL = SELECT Count(*) AS total + this.QueryPart; {j=hQL3
rs = db.executeQuery(SQL); <!HDtN
if (rs.next()) +&zuI
Total = rs.getInt(1); 7Caap/L:
// 设置当前页数和总页数 H2_>Av{m
TPages = (int)Math.ceil((double)this.Total/this.MaxLine); .N><yQ-j3'
CPages = (int)Math.floor((double)Offset/this.MaxLine+1); |`D5XRVbi
// 根据条件判断,取出所需记录 Q@.9wEAJ
if (Total > 0) { _.8]7f`*Gc
SQL = Query + LIMIT + Offset + , + MaxLine; ^l2d?v8
rs = db.executeQuery(SQL); _TcQ12H 5<
} X'Il:SK
return rs; !J?=nSu
} OsSiBb,W79
毫无疑问,这段代码在数据库是Mysql时将会是漂亮的,但是作为一个通用的类(事实上我后面要提供的就是一个通用类库中的一部分),需要适应不同的数据库,而基于这个类(库)的应用,也可能使用不同的数据库,所以,我们将不使用这种方法。 >`V|`Zi ?
AkQFb2|ir
?}Ptb&Vk(
另一种繁琐的实现方法 o?hw2-mH
我看过一些人的做法(事实上包括我在内,一开始也是使用这种方法的),即不使用任何封装,在需要分页的地方,直接操作ResultSet滚到相应的位置,再读取相应数量的记录。其典型代码如下: VKfHN_m*
<% /ykxVCvAt
sqlStmt = sqlCon.createStatement(java.sql.ResultSet.TYPE_SCROLL_INSENSITIVE, {kO:HhUg
java.sql.ResultSet.CONCUR_READ_ONLY); J2k'Ke97o
strSQL = select name,age from test; <W|{)U?p
//执行SQL语句并获取结果集 kX .1#%Ex
sqlRst = sqlStmt.executeQuery(strSQL); b6$A@b
//获取记录总数 9oN'.H^
sqlRst.last(); m3!MHe~t
intRowCount = sqlRst.getRow(); TV>R(D3T/
//记算总页数 8;Bwz RtgT
intPageCount = (intRowCount+intPageSize-1) / intPageSize; `TR9GWU+B
//调整待显示的页码 "uERa(i
if(intPage>intPageCount) intPage = intPageCount; w]YyU5rhS
%> "&o@%){]
<table border=1 cellspacing=0 cellpadding=0> Tu#k+f*s
<tr> 0YRYCO$
<th>姓名</th> _q4dgi z
<th>年龄</th> CbaAnm1
</tr> gY^TBR0?m
<% (S 3kP5:F
if(intPageCount>0){ \yizIo.Y`
//将记录指针定位到待显示页的第一条记录上 MZMv.OeYt,
sqlRst.absolute((intPage-1) * intPageSize + 1); @ y2Bq['
//显示数据 >oYwzK0&
i = 0; ieoUZCO^r\
while(i<intPageSize && !sqlRst.isAfterLast()){ =` >Nfa+,
%> F88SV6
<tr> Pw{{+PBu R
<td><%=sqlRst.getString(1)%></td> @%85k/(
<td><%=sqlRst.getString(2)%></td> Y$5v3E\uc
</tr> Kyiez]T6%q
<% ;8Q?`=a
sqlRst.next(); >d[vHyA~!D
i++; }nERQq&A
} XzFqQ-H
} j|8{Vyqd
%> D(?#oCCA
</table> S5vMP
N
很显然,这种方法没有考虑到代码重用的问题,不仅代码数量巨大,而且在代码需要修改的情况下,将会无所适从。 g
{wPw
j`M<M[C*4N
BnY|t2r
使用Vector进行分页 (&x\,19U$
还见过另一些实现分页的类,是先将所有记录都select出来,然后将ResultSet中的数据都get出来,存入Vector等集合类中,再根据所需分页的大小,页数,定位到相应的位置,读取数据。或者先使用前面提到的两种分页方法,取得所需的页面之后,再存入Vector中。 J3E:r_+
扔开代码的效率不说,单是从程序结构和使用的方便性上讲,就是很糟糕的。比如,这种做法支持的字段类型有限,int, double, String类型还比较好处理,如果碰到Blob, Text等类型,实现起来就很麻烦了。这是一种更不可取的方案。 u+FftgA
aVL%-Il}
xH-k~#
一个新的Pageable接口及其实现 (?wKBUi
很显然,看过上面三种实现方法后,我们对新的分页机制有了一个目标,即:不与具体数据库相关;尽可能做到代码重用;尽可能与原JDBC接口的使用方法保持一致;尽可能高的效率。 *njB
fH'
首先,我们需要提供一个与java.sql.ResultSet向下兼容的接口,把它命名为Pageable,接口定义如下: bv" ({:x
public interface Pageable extends java.sql.ResultSet{ Bm>(m{sX>
/**返回总页数 iEO2Bil]
*/ EB<tX`Wp
int getPageCount(); f3|=T8"t
/**返回当前页的记录条数 Q#bo!]H{t
*/ 2_DtzY:=
int getPageRowsCount(); Q*o4zW
/**返回分页大小 !H.lVA
*/ SvJ8Kl OV
int getPageSize(); E*"E{E7
/**转到指定页
v^E2!X
*/ +a@SdWf
void gotoPage(int page) ; X2 kLbe
/**设置分页大小 z1A-EeT
*/ y`Y}P1y*
void setPageSize(int pageSize); 01w/,r
/**返回总记录行数 $l"(tB7d
*/ HYa!$P3}[
int getRowsCount(); AU\!5+RDB
/** ZWW}r~d{
* 转到当前页的第一条记录 pDN,(Ip
* @exception java.sql.SQLException 异常说明。 #>NZN1
*/ 1S@k=EKM
void pageFirst() throws java.sql.SQLException; (G'ddZAJV
/** ,urkd~
* 转到当前页的最后一条记录 ;!Bkk9r"H
* @exception java.sql.SQLException 异常说明。 5mBk[{
*/ CBHWMetJ*
void pageLast() throws java.sql.SQLException; @isqFKjph
/**返回当前页号 ew~FN
*/ c(JO;=,@9
int getCurPage(); 5n#&Hjb*F0
} D4T+Gk"n
这是一个对java.sql.ResultSet进行了扩展的接口,主要是增加了对分页的支持,如设置分页大小,跳转到某一页,返回总页数等等。 |,f6c
Omf
接着,我们需要实现这个接口,由于这个接口继承自ResultSet,并且它的大部分功能也都和ResultSet原有功能相同,所以这里使用了一个简单的Decorator模式。 B}T72!a
PageableResultSet2的类声明和成员声明如下: l/M+JT~R
public class PageableResultSet2 implements Pageable { g}h0J%s
protected java.sql.ResultSet rs=null; I[ C.iILL
protected int rowsCount; J(L$pIM
protected int pageSize; p 1fnuN |,
protected int curPage; (#BA{9T,^
protected String command = ; 6?~pjMV
} N|d@B{a(
可以看到,在PageableResultSet2中,包含了一个ResultSet的实例(这个实例只是实现了ResultSet接口,事实上它是由各个数据库厂商分别实现的),并且把所有由ResultSet继承来的方法都直接转发给该实例来处理。 %%u4('=
PageableResultSet2中继承自ResultSet的主要方法: LRgk9*@,
//…… 94/}@<d-=
public boolean next() throws SQLException { Dne&YVF9V
return rs.next(); rbWFq|(_
} !qq@F%tv
//…… 1Pc'wfj
public String getString(String columnName) throws SQLException { 7%WI
try { 4 .qjTR
return rs.getString(columnName); _en 8hi@Z
} W`kgYGnFG
catch (SQLException e) {//这里是为了增加一些出错信息的内容便于调试 8i"fhN3?Y
throw new SQLException (e.toString()+ columnName= bZJiubBRI
+columnName+ SQL=+this.getCommand()); Su^Z{ Ud`
} X}
8U-N6)
} $S/ 8T
//…… =="SW"vNi
只有在Pageable接口中新增的方法才需要自己的写方法处理。 uEY5&wX`
/**方法注释可参考Pageable.java tL|Q{+i
yE
*/ -ybupUJcbv
public int getCurPage() { % *hBrjbj
return curPage; 8'62[e|=7[
} Yzz8:n
public int getPageCount() { To95WG7G
if(rowsCount==0) return 0; re2%e-F"
if(pageSize==0) return 1; a!.8^:B&
//calculate PageCount F.9|$g*ip
double tmpD=(double)rowsCount/pageSize; kM@,^`&
int tmpI=(int)tmpD; P n DZi
if(tmpD>tmpI) tmpI++; P*Nl3?T
return tmpI; %-.GyG$i
} "tIx$?I
public int getPageRowsCount() { ,'}ZcN2)
if(pageSize==0) return rowsCount; wz57.e!Me=
if(getRowsCount()==0) return 0; sy?W\(x
if(curPage!=getPageCount()) return pageSize; IuL]V TY
return rowsCount-(getPageCount()-1)*pageSize; u^$ CR
} %8/$CR
public int getPageSize() { x(Z@R\C-a
return pageSize;
=>U~ligu
} 7;V5hul
public int getRowsCount() { "`wq:$R
return rowsCount; 2J5dZYW
}
aY~IS?!;
public void gotoPage(int page) { 'Z[R*Ikzq
if (rs == null) /e,lD)
return; Hqk2W*UTl
if (page < 1) Q*5d~Yr ]R
page = 1; V^D#i(5
if (page > getPageCount()) rkrt.B
page = getPageCount(); *9PQJeyR
int row = (page - 1) * pageSize + 1; 6 s/O\A
try { 3h>Ji1vV
rs.absolute(row); - =Hr|AhE
curPage = page; +(
d2hSIF
} Phczf
catch (java.sql.SQLException e) { f.{0P-Np
} ( KrIMZ
} ~kga+H
public void pageFirst() throws java.sql.SQLException { =
zSrre
int row=(curPage-1)*pageSize+1; Ra5cfkH;
rs.absolute(row); _<$=n6#
} hG U &C]
public void pageLast() throws java.sql.SQLException { ),_bDI L+
int row=(curPage-1)*pageSize+getPageRowsCount(); T/ov0l_
rs.absolute(row); f$/D?q3N
} w>eOERZa
public void setPageSize(int pageSize) { okW3V}/x/z
if(pageSize>=0){ iT5%X
this.pageSize=pageSize; A@4Cfb@
curPage=1; ~Hq
2'
} l #Tm`br
} r]yq
#T`z
PageableResultSet2的构造方法: ,^(T^ -
public PageableResultSet2(java.sql.ResultSet rs) throws java.sql.SQLException { Hcpw[%(
if(rs==null) throw new SQLException(given ResultSet is NULL,user); K|&