foxty [原作]
近日一直在研究如何寫出高小的分頁演算法,大概整理了一下,思路如下:
首先資料庫裡需要有一個自動編號欄位(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.downcodes.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,將資料增加到128,000筆記錄,舊程式存取一頁需要1900毫秒左右,新程式第一次存取需要2300毫秒左右,然後每次存取只需要70毫秒左右。
這裡要注意的是資料庫每改動一次,Application("PageId") 就需要重新賦值!
研究心得:(首先謝謝葉子(DVBBS)的心得)盡量不要用自帶的分頁程序,Rs.RecordCount 很耗資源。依次,估計Rs.PageCount …也耗資源,而且用Rs.GetRows()效果也很明顯提高。
經過比較,葉子的演算法在記錄比較靠前的時候速度以及效率是比較高的。但是不太穩定,有時(很少)會從30毫秒左右跳到1-200毫秒。到了後面效率就明顯下降到50-80毫秒,越後效率越低。新演算法第一次效率比較低下,大約在500毫秒左右,但是比較穩定,後面一般哦度是50毫秒左右,而且隨著函式庫的記錄數變化,這個速度依然如此。不會有什麼變化。下次就把葉子和我的演算法結合起來試試,不過葉子的演算法確實很不錯D,具備通用性。我這個只能拿來聊聊了。