在Asp中如何快速优化分页的技巧

本文介绍了一种高效的数据库分页算法,通过预计算每页的ID范围并存储在Application中,实现快速检索,显著提升了大数据量下的分页性能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

近日一直在研究如何才能写出高小的分页算法,大概整理了一下,思路如下:

 

    首先数据库里需要有一个自动编号字段(ID)。然后第一次访问的时候,取出所有记录,定制好每页的记录数PageSize,计算出页数,然后根据页数建立一个一维数组PageId(PageCount),PageId(0)保存记录初试条件,然后对应每个元素保存每页对应的ID边界码

(

  1,ID边界码:如果数据库记录ID记录序列如下  1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16

  假设需要按照ID 顺序排序的话 ,PageSize = 5, Pagecount = 4 ,PageId(4)

   数组PageId的值分别为PageId(0) = 1, PageId(1) = 5 ,PageId(2) = 10,PageId(3) = 15 ,PageId(4) = 16

   当访问第 i 页的时候就直接找 [PageId(i-1) , PageId(i) ) 之间的记录,这样可以保证每次取的记录都只是PageSize 条记录。 

  假设需要按照ID倒序排列的话,

   数组PageId的值分别为PageId(0) = 16 , PageId(1) = 12 , PageId(2) = 7 ,PageId(3) = 2, PageId(4) = 1, 当访问第 i 页的时候就直接查找ID属于[ PageId(i-1) , PageId(i) )  



)

 将数组PageId()保存在Application()中,以便访问,这样,只是第一次访问分页程序的时候便初始化Application()。代码部分如下:(下面称为新程序)



<%

 Time1 = Timer()

 Dim Conn

 Set Conn = Server.CreateObject("Adodb.Connection")

 Conn.open "Driver={MicroSoft Access Driver (*.mdb)};Dbq="&Server.MapPath("db.mdb")

 'www.knowsky.com

 Dim Page,PageCounts,PageId,PageList

 Dim Rs,Sql

 Dim IsInit,i

 

 IsInit   = False                                         '标志为,用来判断Application("PageId")是否初始化

 PageList = 20                                            '设置每页显示20条数据

 Set Rs    = Server.CreateObject("Adodb.Recordset")

 Page     = Request.QueryString("Page")                         '注意页码需要检查类型 

  

 If IsEmpty(Application("PageId")) Then               '如果Application("PageId")还未初始化,则先进行初始化

  Response.Write("Init app!<br>")

  Sql      = "Select * From test Order By Id Desc"  '假定这里是按照ID倒序排列

  Rs.open Sql,Conn,1,1  '得到记录集对象

  

  If Not (Rs.Eof  or Rs.Bof) Then 

   Rs.PageSize = PageList                        '设置每页记录数

   PageCounts  = Rs.PageCount

   ReDim PageId(PageCounts)                      '重新定义数组PageId

   For i = 0 To PageCounts                       '开始给数组 PageId() 赋值    

    If Rs.eof Then Exit For

    PageId(i) = Rs("ID")

    Rs.Move (PageList)

   Next

   Rs.MoveLast

   PageId(PageCounts) = Rs("ID")

   Application.Lock()

   Application("PageId") = PageId

   Application.UnLock()  

  End If

  Rs.Close

 End If

 IdStart = Clng(Application("PageId")(Page-1))

 IdEnd   = Clng(Application("PageId")(Page))

 Sql = "Select * from test where id<="&IdStart&" and id>"&IdEnd&" "

 Rs.open Sql,Conn,1,1

 While Not Rs.eof

  Response.Write(rs(0)&"--"&rs(1)) 

  Rs.MoveNext

 Wend

 Rs.Close  

 Set Rs = Nothing

 Conn.Close

 Set Conn = Nothing

 

 

 For i = 1 To Ubound(Application("PageId"))

  Response.Write("<a href='Test1.asp?Page="&i&"'>"&i&"</a> ")

 Next

 Time2 = Timer()

 

 Response.Write("<br>"&(Time2-Time1)*1000) 

 'Application.Contents.Remove("PageId")

%> 



传统分页代码如下:(下面称为旧程序)

<%

 Time1 = Timer()

 Dim Conn

 Set Conn = Server.CreateObject("Adodb.Connection")

 Conn.open "Driver={MicroSoft Access Driver (*.mdb)};Dbq="&Server.MapPath("db.mdb")

 

 Dim Page,PageCounts,PageList

 Dim Rs,Sql

 

 PageList = 20

 Page     = Request.QueryString( "Page" )

 Set Rs   = Server.CreateObject("Adodb.Recordset")

 Sql      = "Select * from test order by id desc"

 Rs.Open Sql,Conn,1,1

 

 If Page = "" Then Page = 1

 If Not( Rs.eof Or Rs.Bof ) Then

  Rs.PageSize     = PageList

  PageCounts      = Rs.PageCount

  Rs.AbsolutePage = Page  

 End If

 

 For i = 1 to PageList

  If Rs.eof Then Exit For

  Response.Write(Rs(0)&"-----"&Rs(1)&"<br>")

  Rs.MoveNext

 next

 

 For i = 1 To PageCounts

  Response.Write("<a href='Test.asp?Page="&i&"'>"&i&"</a> ")

 Next

 Time2 = Timer()

 

 Response.Write("<br>"&(Time2-Time1)*1000) 

%>

 

    其实,总体的思想就是,建立一个Application("PageId")全局数组,每个元素都保存页面所区记录的ID区间,比如,Application("PageId")(0) 保存第一个元素的ID,然后Application("PageId")(1)保存下一页的第一个ID…………依次类推,当需要访问第 i 页的时候,就直接查找ID在 [ Application("PageId")(i-1)  , Application("i") ) 里面的记录集,这样,每次只用查找需要的记录数,而不需要每次都把所有记录都查找一遍,但是,这个方法是在第一次访问的时候,即需要创建数组Application("PageId")的时候比较慢一点,当第N次访问的时候 (N>1)速度就快将近10倍,我采用上面2个程序测试:

  1,数据库记录有32000条记录,旧程序访问一页需要500毫秒左右,新程序只是第一次访问的时候达到这个时间,然后每次都只需要55毫秒左右。

  2,将数据增加到64000条记录,旧程序访问一页需要1000毫秒左右,新程序也是第一次访问的时候达到这个似乎件, 后面每次仍然还是保持在55毫秒左右。

  3,将数据增加到128000条记录,旧程序访问一页需要1900毫秒左右,新程序第一次访问需要2300毫秒左右,然后每次访问只需要70毫秒左右。

  这里需要注意的是数据库每改动一次,Application("PageId") 就需要重新赋值!



    研究心得:(首先谢谢叶子(DVBBS)的心得)尽量不要用自带的分页程序,Rs.RecordCount 很耗资源。依次,估计Rs.PageCount ……也耗资源,而且用Rs.GetRows()效果也很明显提高。

 

    经过比较,叶子的算法在记录比较靠前的时候速度以及效率是比较高的。但是不太稳定,有时(很少)会从30毫秒左右跳到1-200毫秒。到了后面效率就明显下降到50-80毫秒,越后效率越低。新算法第一次效率比较低下,大约在500毫秒左右,但是比较稳定,后面一般哦度是50毫秒左右,而且随着库的记录数变化,这个速度依然如此。不会有什么变化。下次就把叶子和我的算法结合起来试试,不过叶子的算法确实是很不错D,具备通用性。我这个只能拿来聊聊了。



在网上找了一个 Asp数据操作组件(百万级分页) ,实用环境:Asp+DLL+MsSQL(这个网上很多地方都可以下载),并且其中结合SQL存储过程,说实话以前SQL存储过程接触的比较少,所以在调试该组件的时候遇到了一些问题,并试着改了一下(作者别生气(^..^)),高手们别笑话我。 原代码如下: 有分页列表数据: cls.tblName="Tablename" '表名 cls.fldName="Id" '排序关键字段 cls.PageIndex=Request("P")*1 '当前页码 cls.PageSize=20 '每页列数 cls.OrderType=1 '排序方式,0 、1 cls.strWhere=StrW '查询条件,不带Where cls.ListUrl="?Y="&Y&"&M="&M&"&D="&D&"&Sn="&Sn&"&P=" 分页URL Set Rs=cls.Result do while not rs.eof rs.movenext loop 分页数据 cls.page 在原文的使用说明中,在翻页代码中对表查询默认是全部字段也就是*,这样是不是会浪费资源,而无法查询需要的字段,于是我在DLL代码中新加入一个变量名为zdName,作为传递可控制查询条件的变量,同时在SQL存储过程中也加入相应的接受变量@zdName 字符型。 在这个小问题解决后,又在使用上发现cls.strWhere付给它的变量StrW 如果条件变量其中带有类似 例如 a='b'则执行SQL存储过程时候会提示错误,后来发现是DLL代码中: sql = "exec GetList " & tblName & "," & fldName & "," & PageSize & "," & PageIndex & "," & OrderType & ",'" & strWhere & "' " 这个地方是否写的不够严谨,在StrW中如果含有单引,那么将无法执行,所以我在DLL的VB代码改成 If InStr(strWhere, "'") 0 Then sql = "exec GetList " & tblName & "," & fldName & "," & PageSize & "," & PageIndex & "," & OrderType & ",""" & strWhere & """ " Else sql = "exec GetList " & tblName & "," & fldName & "," & PageSize & "," & PageIndex & "," & OrderType & ",'" & strWhere & "' " End If 试了试目前的问题确实解决了,这个组件我也是刚刚使用,希望对大家能有点帮助!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值