前言 >f_D|;EV
在使用数据库的过程中,不可避免的需要使用到分页的功能,可是JDBC的规范对此却没有很好的解决。对于这个需求很多朋友都有自己的解决方案,比如使用Vector等集合类先保存取出的数据再分页。但这种方法的可用性很差,与JDBC本身的接口完全不同,对不同类型的字段的支持也不好。这里提供了一种与JDBC兼容性非常好的方案。 ,@<-h* m
JDBC和分页 ,,@_r&f:
Sun的JDBC规范的制定,有时很让人哭笑不得,在JDBC1.0中,对于一个结果集(ResultSet)你甚至只能执行next()操作,而无法让其向后滚动,这就直接导致在只执行一次SQL查询的情况下无法获得结果集的大小。所以,如果你使用的是JDBC1.0的驱动,那么是几乎无法实现分页的。 +|o-lb
好在Sun的JDBC2规范中很好的弥补了这一个不足,增加了结果集的前后滚动操作,虽然仍然不能直接支持分页,但我们已经可以在这个基础上写出自己的可支持分页的ResultSet了。 ysL8w"t
hzPpw.
hR. EZ|.
PUa~Apj'
和具体数据库相关的实现方法 |=7%Edkd
有一些数据库,如Mysql, Oracle等有自己的分页方法,比如Mysql可以使用limit子句,Oracle可以使用ROWNUM来限制结果集的大小和起始位置。这里以Mysql为例,其典型代码如下: #'"h+[XY
// 计算总的记录条数 |Q7Ch]G
String SQL = SELECT Count(*) AS total + this.QueryPart; (s}9N
rs = db.executeQuery(SQL); *A_
if (rs.next()) A@`C<O ^
Total = rs.getInt(1); @GGyiK@
// 设置当前页数和总页数 ~r!j VK>^
TPages = (int)Math.ceil((double)this.Total/this.MaxLine); $-o 39A#
CPages = (int)Math.floor((double)Offset/this.MaxLine+1); G"J6X e
// 根据条件判断,取出所需记录
I2zSoQ1P
if (Total > 0) { Jq.26I=
SQL = Query + LIMIT + Offset + , + MaxLine; #{N#yReh
rs = db.executeQuery(SQL); \Z)'':},C
} u |#ruFR
return rs; vnIxI a
} "i#!
毫无疑问,这段代码在数据库是Mysql时将会是漂亮的,但是作为一个通用的类(事实上我后面要提供的就是一个通用类库中的一部分),需要适应不同的数据库,而基于这个类(库)的应用,也可能使用不同的数据库,所以,我们将不使用这种方法。 FxD" z3D
z.{yVQE
b5yb~;0
另一种繁琐的实现方法 );=JoRQ{
我看过一些人的做法(事实上包括我在内,一开始也是使用这种方法的),即不使用任何封装,在需要分页的地方,直接操作ResultSet滚到相应的位置,再读取相应数量的记录。其典型代码如下: }p&aI?-B
<% |4dNi1{Zd
sqlStmt = sqlCon.createStatement(java.sql.ResultSet.TYPE_SCROLL_INSENSITIVE, Ef7Kx49I
java.sql.ResultSet.CONCUR_READ_ONLY); 654PW9{(
strSQL = select name,age from test; Z3[,Xw
//执行SQL语句并获取结果集 D@\97t+
sqlRst = sqlStmt.executeQuery(strSQL); o6{XT.z5qx
//获取记录总数 c5Offnq'1
sqlRst.last(); {\ .2h
intRowCount = sqlRst.getRow(); 2b !b-
//记算总页数 ZW,PZ<
intPageCount = (intRowCount+intPageSize-1) / intPageSize; _&yQW&vH#
//调整待显示的页码 QAu^]1 ;
if(intPage>intPageCount) intPage = intPageCount; k"AY7vq@!P
%> 'X`\vTxB
<table border=1 cellspacing=0 cellpadding=0> hI/p9
`w
<tr> uE/qraA
<th>姓名</th> g|2D(J
<th>年龄</th> _)^(-}(_D
</tr> 6W3}6p
<% .%D] z{''
if(intPageCount>0){ FSH6C2
//将记录指针定位到待显示页的第一条记录上 !M}&dW2
sqlRst.absolute((intPage-1) * intPageSize + 1); _Hkc<j/e~
//显示数据 =#1/<q)L
i = 0; po{f*}gas]
while(i<intPageSize && !sqlRst.isAfterLast()){ ?t<wp3bZ
%> W/J3sAYv
<tr> q^,^tw
<td><%=sqlRst.getString(1)%></td> UY>{e>/H9
<td><%=sqlRst.getString(2)%></td> 78 3a Z8
</tr> ,/Xxj\i
<%
E?%k
sqlRst.next(); 'zRd?Z>%
i++; w}7`Vas9
} SU x\qz)
} *6k
(xL
%> c?wFEADn
</table> Kz 'W
|
很显然,这种方法没有考虑到代码重用的问题,不仅代码数量巨大,而且在代码需要修改的情况下,将会无所适从。 ujDAs%6MZ
S,J'Z:spf
M~3(4,
使用Vector进行分页 |n]^gTJt
还见过另一些实现分页的类,是先将所有记录都select出来,然后将ResultSet中的数据都get出来,存入Vector等集合类中,再根据所需分页的大小,页数,定位到相应的位置,读取数据。或者先使用前面提到的两种分页方法,取得所需的页面之后,再存入Vector中。 oq;}q
扔开代码的效率不说,单是从程序结构和使用的方便性上讲,就是很糟糕的。比如,这种做法支持的字段类型有限,int, double, String类型还比较好处理,如果碰到Blob, Text等类型,实现起来就很麻烦了。这是一种更不可取的方案。 tXfB.[U
{K:/(\
|" l
g4S%
一个新的Pageable接口及其实现 hXYVi6(k
很显然,看过上面三种实现方法后,我们对新的分页机制有了一个目标,即:不与具体数据库相关;尽可能做到代码重用;尽可能与原JDBC接口的使用方法保持一致;尽可能高的效率。 <;W4Th<4
首先,我们需要提供一个与java.sql.ResultSet向下兼容的接口,把它命名为Pageable,接口定义如下: b/<4\f
public interface Pageable extends java.sql.ResultSet{ en#W<"_"
/**返回总页数 &
yw-y4 =
*/ =axi0q?}
int getPageCount(); S0kH/A
/**返回当前页的记录条数 [_b10Z'{
*/ SkN^ytKE
int getPageRowsCount(); E6BW&Xp
/**返回分页大小 vUj7rDT|
*/ !$Mv)c/_u
int getPageSize(); ];oED?I
/**转到指定页 w/Ia`Tx$
*/ drF"kTD"7
void gotoPage(int page) ; \$9S_z
/**设置分页大小 V8&%f xn+
*/ wwE9|'Ok
void setPageSize(int pageSize); /&vUi7'
/**返回总记录行数 C$rZn%dp(
*/ w)3LY F
int getRowsCount(); w=O:|Xu#*
/** n j1 cqh
* 转到当前页的第一条记录 mnG\UK,k
* @exception java.sql.SQLException 异常说明。 RkC?(p
*/ .bew,92
void pageFirst() throws java.sql.SQLException; &XN*T.Y`
/** [NC^v.[1[
* 转到当前页的最后一条记录 \5X34'7
* @exception java.sql.SQLException 异常说明。 {9Y@?
*/ ]+,Z()
void pageLast() throws java.sql.SQLException; 5tQffo8t
/**返回当前页号 >e8t
*/ u!;kBs
int getCurPage(); #F[6$. Gr
} Cc9<ABv?
这是一个对java.sql.ResultSet进行了扩展的接口,主要是增加了对分页的支持,如设置分页大小,跳转到某一页,返回总页数等等。 Bg;bBA!L
接着,我们需要实现这个接口,由于这个接口继承自ResultSet,并且它的大部分功能也都和ResultSet原有功能相同,所以这里使用了一个简单的Decorator模式。 Qb9) 1
PageableResultSet2的类声明和成员声明如下: |A &Nv~.)
public class PageableResultSet2 implements Pageable { /V:%}Z
protected java.sql.ResultSet rs=null; &zUo", }9
protected int rowsCount; \:^$ZBQr<n
protected int pageSize; W._vikR
protected int curPage; ])0&el3-
protected String command = ; XWk/S $-d
} ^8E/I]-
可以看到,在PageableResultSet2中,包含了一个ResultSet的实例(这个实例只是实现了ResultSet接口,事实上它是由各个数据库厂商分别实现的),并且把所有由ResultSet继承来的方法都直接转发给该实例来处理。 t+p-,ey^@
PageableResultSet2中继承自ResultSet的主要方法: )p ,-TtV
//…… '[:].?M
public boolean next() throws SQLException { /C_O/N
return rs.next(); 9?l(
}S`
} "'s`?
//…… `7+?1z
public String getString(String columnName) throws SQLException { ^ !E;+o' t
try { p8o%H-Xk
return rs.getString(columnName); W:hR81ci
} ,O $F`0>9A
catch (SQLException e) {//这里是为了增加一些出错信息的内容便于调试 \-`L}$
throw new SQLException (e.toString()+ columnName= A#LK2II^
+columnName+ SQL=+this.getCommand());
9JP{F
} Vs\)w>JF
} r'w5i1C+
//…… EtN"K-X
只有在Pageable接口中新增的方法才需要自己的写方法处理。 `Fu|50_@V
/**方法注释可参考Pageable.java 3p$ZHH.UP
*/ cb|`)"<HN
public int getCurPage() { K)@]vw/\
return curPage; H;Z{R@kf
} CM8WI~
public int getPageCount() { i8u9~F
if(rowsCount==0) return 0; G8f7N;D
if(pageSize==0) return 1; rTW1'@E
//calculate PageCount [ZDJs`h!`
double tmpD=(double)rowsCount/pageSize; I3s'44
int tmpI=(int)tmpD; i1 C]bUXA
if(tmpD>tmpI) tmpI++; I-&/]<5y
return tmpI; Lp1wA*
} RhX
2qsva-
public int getPageRowsCount() { TDy@Y>
)
if(pageSize==0) return rowsCount; li,kW`j+t
if(getRowsCount()==0) return 0; eAm7*2
if(curPage!=getPageCount()) return pageSize; &Lk@Xq1
return rowsCount-(getPageCount()-1)*pageSize; Sg')w1
} 32YE%
public int getPageSize() { {tF=c0Z
return pageSize; e7pN9tXGf
} B_c(3n-"
public int getRowsCount() { g 9>p?XY
return rowsCount; &> }MoB
} W $H8[G
public void gotoPage(int page) { ]N2'L!4|;
if (rs == null) `[57U,v
return; ;,@3bu>r
if (page < 1) ]-L/Of6F)|
page = 1; B~yD4^
if (page > getPageCount()) Qh?q0VKU^
page = getPageCount(); s13Iu#
int row = (page - 1) * pageSize + 1; $?ke "
try { 6L'cD1pu
rs.absolute(row); :8yrtbf$
curPage = page; Kxh)'aal
} ,&z_ 2m
catch (java.sql.SQLException e) { ,7>_Lp_v
} _mA[^G=gY
} K31Fp;K
public void pageFirst() throws java.sql.SQLException { -V_e=Y<J/
int row=(curPage-1)*pageSize+1; >L[,.}(9
rs.absolute(row); QF!K$?EU[
} *l_1T4]S
public void pageLast() throws java.sql.SQLException { 2Np9*[C
int row=(curPage-1)*pageSize+getPageRowsCount(); 0z.`
rs.absolute(row); x/bO;9E%U4
} AUzJ:([V
public void setPageSize(int pageSize) { q'",70"\
if(pageSize>=0){ B RD>q4w
this.pageSize=pageSize; nLdI>c9R
curPage=1; @fbvu_-].
} r{p?aG
} BYNOgB1
PageableResultSet2的构造方法: )1lYfJ
public PageableResultSet2(java.sql.ResultSet rs) throws java.sql.SQLException { q\d'}:kfu
if(rs==null) throw new SQLException(given ResultSet is NULL,user); &'T7 ~M:
''v_8sv
rs.last(); o6Vc}jRH
rowsCount=rs.getRow(); )<-kS
rs.beforeFirst(); 'Kp|\Tr
@2kt6
W
this.rs=rs; :m@(S6T m
} $o{f)'.>n
这里只是简单的取得一个总记录数,并将记录游标移回初始位置(before first),同时将参数中的ResultSet赋给成员变量。 (O/hu3
Kgk9p`C(
3P I{LU
Pageable的使用方法 f^m8 4o'
因为Pageable接口继承自ResultSet,所以在使用方法上与ResultSet一致,尤其是在不需要分页功能的时候,可以直接当成ResultSet使用。而在需要分页时,只需要简单的setPageSize, gotoPage,即可。 VUagZ7p
PreparedStatement pstmt=null; sN^R Z0!>
Pageable rs=null; 4Q_2GiF_
?
……//构造SQL,并准备一个pstmt. A -c3B+
rs=new PageableResultSet2(pstmt.executeQuery());//构造一个Pageable p.8G]pS
rs.setPageSize(20);//每页20个记录 qhL e[[>
rs.gotoPage(2);//跳转到第2页 wyvs#T
for(int i=0; i<rs.getPageRowsCount(); i++){//循环处理 6i=m1Yk
int id=rs.getInt(“ID”); ?%*Zgk!l7
……//继续处理 +!.=M8[
} > YN<~z-
Tet,mzVuu
YNk?1#k?i
总结 ?Za1
b
一个好的基础类应该是便于使用,并且具备足够的可移植性,同时要保证其功能的完善。在上面的实现中,我们从java.sql.ResultSet接口继承出Pageable,并实现了它。这就保证了在使用中与JDBC原有操作的一致性,同时对原有功能没有缩减。 L{<E'#@F
同时它也是易于使用的,因为封装了一切必要的操作,所以在你的代码中唯一显得难看和不舒服的地方就是需要自己去构造一个PageableResultSet2。不过只要你愿意,这也是可以解决的。 "1h|1'S50?
当然它也有具有充分的可移植性,当你将数据库由Oracle变为Mysql或者SQLServer的时候,你仍然可以使用这些分页的代码。它在使用中(或者说在移植的过程中)唯一的限制就是你必须要使用一个支持JDBC2的驱动(现在明白为什么我把类命名为PageableResultSet2了吧。:P),不过,好在JDBC2已经成为标准了,绝大多数的数据库(如Oracle, Mysql, SQLServer)都有自己的或者第三方提供的JDBC2的驱动。 |]\qI
OK,这个分页的实现是否对你的编程有帮助呢?仔细看看,其实真正自己写的代码并不多的,大部分都只是简单的转发操作。一个合适的模式应用可以帮你很大忙。