ASP緩存技術詳解

一、何謂ASP緩存/爲什麼要緩存
  當你的web站點採用asp技術建立的初期,可能感覺到的是 asp動態網頁技術帶來的便利性,以及隨意修改性、自如的http控制。但是,隨着訪問量的增加,你一定會發現自己的站點訪問速度越來越慢,IIS重新啓動得越來越頻繁。接下來,你一定想優化asp,諸如更換性能更優異的數據庫、建立索引、編寫存儲過程等等。這些措施有些不需要增加成本壓力,有些則成本壓力很大(譬如叢access到SQL),而且效果還不一定。

  面對web訪問壓力,我認爲最經濟的辦法是利用緩存優化技術來實現緩解 web的服務壓力。
Web訪問量增加通常意味着以下資源需求的快速增長:

1、 網卡流量增加,需要消耗更多的CPU處理網絡流量和網絡I/O線程。
2、 需要更頻繁的打開/關閉數據庫連結(如果使用數據庫技術—通常asp都會採用數據庫作爲數據存儲),嚴重消耗資源的事物數量、以及事務相互競爭資源引起的死鎖、會增加網絡I/O或者CPU消耗。
3、 如果採用session的話,IIS爲了維持狀態,會消耗更多內存,而內存消耗可能會引發物理內存不夠,引起物理內存同輔存間的頻繁交換,從而引起代碼執行的停頓,web響應阻滯。
4、 由於訪問的不到及時響應,會引起網頁訪問故障,導致用戶刷新,從而加劇CPU、內存等資源需求。

  實際上,考慮通常的web應用程序,很多時候的動態代碼執行是不必要的。

二、asp緩存的分類

  擅自總結,asp的緩存可以分爲兩類:

  1、 文件緩存
  所謂文件緩存,就是根據邏輯判斷,一段時間內某個asp的特定執行將不會有很大的變動,因而將內容以靜態html的形式存放,然後以web的重定向技術讓客戶端訪問靜態文件,以達到減少CPU、數據庫資源等的需求。這樣的應用很多,譬如很多論壇就是在回覆貼子的時候將整個貼子重新生成一個靜態文件,然後進行重定向的,例如 donews.com的論壇。該成靜態還有一個副作用(好處)–可以很容易被google等搜索引擎收錄。一些所謂新聞發佈系統的都採用了此技術。

  2、 文件片斷緩存

  所謂文件緩存,也是基於邏輯判斷,某部分數據(通常是需要消耗資源的大容量數據庫查詢取得)在一定時間內不會改變,所以我們可以將這些數據利用文件的形式進行存儲,當需要時候,可以通過讀取文件來獲取數據,避免增加數據庫的負擔。例如,我們通常將一些數據以xml格式存儲,然後利用xslt技術實現顯示(xml處理通常需要大量CPU資源,所以通常是IE直接讀取xml到客戶端在客戶的CPU上進行處理)。CSDN的論壇就是這樣處理的。

  3、 主存緩存

  除此之外,還可以考慮在內存中處理緩存,將需要及時響應的內容存儲在內存中,一旦訪問需求,立即從快速的貯存中輸送出去。如果極大量的訪問需求集中在幾個少量的頁面或者主存足夠多,我想採用主存緩存一定可以大幅度提高web訪問性能。

三、如何實現/使用緩存

  實現緩存需要考慮以下問題:

  1、 哪些頁面會在短時間內不會改變?
  分析自己的站點,這樣的頁面很多。譬如一個站點通常都有新聞資訊類的欄目,這些欄目通常都是站點維護人員在一天的某個時間發佈資料,之後很少改動頁面。那麼這些頁面就適合於採用靜態文件緩存。實際上,所謂新聞發佈系統就是這麼做的,那麼那也可以參考這些系統的思想改造自己的原有動態asp頁面。

  2、 那些頁面針對全部訪問者都採用同一個邏輯生成(也就是不區分訪問者)。
  除了新聞資訊之類的欄目所有訪問者都看一個界面外,論壇等消耗資源的應用一般也可以設計成統一邏輯生成(同一個貼子,張三李四看的都一樣),針對這類應用頁面我們也可以採用靜態緩存來實現。也可以考慮將數據片斷化,利用腳本技術在服務器處理能力之外也就是客戶端瀏覽器進行處理。

  3、 採用緩存的代價和收穫。
  主要就是“空間換(響應)時間”。利用緩存技術將之後頻繁需要的內容進行預處理,使之提高web服務器響應能力,更重要贏得訪問者的歡心。
代價就是web空間需求增加,同時又可能影響到訪問效果。
但我認爲適當的緩存,是利大於弊的。

  4、 那些地方不適宜採用緩存
  動態查詢頁面,每個人的查詢內容不一樣,所以顯示結果不大一樣,所以不大可能將查詢結果生成緩存,所以採用緩存較爲複雜且緩存利用率底下,造成管理成本上什(假設你緩存了1000個查詢關鍵字,那麼管理這些關鍵字同緩存的對應也是麻煩事)。

四、實例分析

  假設一個建議論壇的原有佈局如下:

根目錄下:
default.asp 首頁,一般是精華、推薦之類
listBorad.asp 該文件列出全部分欄目的名稱和介紹,如果攜帶參數MainBID就表示要列出板塊下的欄目
listThread.asp 該文件如果不攜帶任何參數表示列出全部的貼子,攜帶MainBID表示列出某塊的全部貼子。如果攜帶subBID表示列出具體欄目的貼子。如果攜帶 page參數表示分頁列出主題。
ViewThread.asp 列出某個貼子內容。我們假設貼子顯示爲一個發言,任意跟貼全部列在後面。ID參數爲要顯示的貼子。
Reply.asp 迴應某個貼子,攜帶參數Id迴應某個貼子

  其它的暫不討論。

  以上,我們可以看到,如果全部是採用原始的 ASP/PHP來做,那幾乎每一個asp文件的執行都需要數據庫操作,頻繁的查詢,多表查詢。要知道查詢數據庫最終會帶來性能的下降,響應速度下降,帶給訪問者緩慢的瀏覽影響,不利於web的質量。更重要的是對於甲乙兩個人來將,他們訪問ViewThread.asp之類的如果ID一致,那麼很多時候他們會看到同樣的內容(他們的瀏覽器收到的HTML代碼幾乎一樣),但是爲了這“同樣的內容”,服務器需要打開數據庫連結、查詢,讀取紀錄,顯示,關閉紀錄、數據庫連結。。。。以下列的消耗服務器資源的操作,如果是更多的人來訪問,最終的結果是這些人加劇消耗服務器資源。實際上,這些爲了“同樣的內容”所做的重複勞動是可以利用緩存技術進行優化避免的。譬如:

  在reply.asp提交內容後,我們立即調用生成靜態的功能,將整個貼子內容存儲爲viewThread_xxxx.htm之類的靜態html文件,再通常情況下訪問viewThread.asp?ID=xxxx的時候,系統自動 redirect到對應的靜態文件viewThreadxxxx.htm去。這樣,當一個貼子沒有最新發布時候,他始終是靜態內容提供給瀏覽者;一旦有了新的提交,將會更新到靜態文件中去,這樣,將會節省很多次數據庫操作,大大提高響應速度。

  listBorad.asp也可以實施靜態化。我們可以分析其可能攜帶的參數,將緩存文件名設定爲listBoard_xx.htm,在增加新的欄目時候進行更新 listBoard_xxx.htm。listThread.asp也類似,只不過由於其參數更多,所以緩存文件也會很多。擊若要緩存 listThread.asp? subBID=xxx&page=2,那麼對應的靜態文件是listThread_xxx_p2.htm。default.asp也一樣。

  那麼如何判斷什麼時候更新?在什麼時機更新?
討論listThread.asp? subBID=xxx&page=2,我們在執行listThread.asp俄時候提取subID和page,然後探測 listThread_xxx_p2.htm是否存在,如果不存在就調用靜態生成功能進行生成該文件,最終重定向到此靜態文件。注意,此處的不存在就意味着出現了新的內容需要我們進行更新。

  那如何造成文件不存在呢?刪除。我們在發表一個新的貼子、刪除貼子、移動貼子的時候我們可以將類似listThread_xxx_p2.htm之類的靜態文件全部刪除。這樣就通知了何時要進行緩存。

  現在還剩下一個問題,如何生成靜態文件?

  我們注意到,之前我們提到的“同樣的內容”。我們可以將改造前的default.asp、listThread.asp等拷貝一個副本,取名爲default_d.asp、listThread_2.asp,且在同一個目錄中(理論上 listThtrad.asp?subID=123同LISTtHREAD_D.ASP?SUBID=123的訪問結果會是同樣的內容),這樣我們在需要生成靜態文件的邏輯中,通過WEB訪問請求的方式調用改造前的副本,得到html代碼,並存儲爲靜態文件。這個web請求實際上相當於在任何真實瀏覽者訪問靜態內容之前,由服務器自身現察看將會輸出的html,然後返回這些代碼,利用文件操作功能存儲爲靜態文件。這樣,緩存文件就在真正瀏覽者之前被創建。

  這樣的方案几乎不觸動原來的佈局,幾乎不會造成因爲改造出現404之類的錯誤。其次,靜態文件也會幫助你的站點容易被被google之類的搜索引擎收錄。何樂而不爲?

  最後,提醒,通過web訪問,asp編程環境下,很多人採用xmlHTTP組件訪問,這會造成很多問題。 xmlhttp自身會cache請求的資源,導致我們通過此組件請求得到的內容不是最新的,造成邏輯上的混亂。所以,應當選擇xml Server http對象或者winhttp組件來實現web請求資源。
使用ASP中的緩存技術可以很大程度上提高你的網站性能,其實這些實現方法是非常的簡單,它將說明如何在服務器上的緩存是如何工作以及你如何使用一種被稱爲斷開連接的ADO連接技術。

  在介紹這些技術之前先說明一下到底什麼是ASP的緩存技術。

  所謂緩存其實就是在內存中開闢一個用來保存數據的空間,使用緩存你就不用頻繁的訪問你保存在硬盤上的數據了,靈活的使用緩存你就免去了心疼的看着可憐的硬盤飽受讀數據時的折磨了。當你一旦執行了一個查詢動作,並且將查詢結果放入緩存中後,你就可以很迅速的重複訪問這些數據了。而如果你不把數據放入緩存的話,當你再次執行這個查詢時,服務器會將進程耗費在從數據庫中獲取並排序上了。

  當數據保存在緩存中時,再次查詢時耗費的時間主要是在顯示數據的時間上了。
也就是說,我們不應該把經常需要改變的數據放到服務端的緩存中,我們應該把改變少,但是又需要經常訪問的數據放到緩存中。

  現在我們先討論ASP在服務端使用緩存的技術,過會再討論ASP如何在客戶端使用
緩存的技術。

  當你有大量的數據(靜態的,就是說變動比較少的)需要顯示給客戶端時,你就可以考慮使用服務端的緩存技術了。這種技術尤其適用於那些顯示風格一致性比較強的網站(呵呵,對於非主流的網站可不好用的說。)

  其實實現方法特別的簡單,大家只要看看下面這個簡單的例子就明白了。

  這是一個用來顯示書籍分類的例子程序

 

DisplayBooks.ASP文件:


< %@ LANGUAGE=JavaScript % >    
< html >    
< body >    
< form method=post >    
書籍分類; < %= getBooksListBox() % >    
< p>    
< input type=submit >    
 
 < %    
function getBooksListBox()    
{    
BooksListBox = Application("BooksListBox")    
if (BooksListBox != null) return BooksListBox;    
crlf = String.fromCharCode(13, 10)    
BooksListBox = "< select name=Books>" + crlf;    
SQL = "Select * FROM Books orDER BY Name";    
cnnBooks = Server.CreateObject("ADODB.Connection");    
cnnBooks.Open("Books", "Admin","");    
rstBooks = cnnBooks.Execute(SQL);    
fldBookName = rstBooks("BookName");    
while (!rstBooks.EOF){    
BooksListBox = BooksListBox + " < option>" +    
fldBookName + "" + crlf;    
rstBooks.MoveNext();    
}    
BooksListBox = BooksListBox + ""    
Application("BooksListBox") = BooksListBox    
return BooksListBox;    
}    
% >
 

很簡單把,其實就是用了很簡單的Application技術,而且就一句話的不同:


 Application("BooksListBox") = BooksListBox
 

你可以驗證一下你就會發現服務器上的請求數量會降低不少的。這種情況尤其適合與那些更新不是很頻繁的網站內容,例如你一天(或則很長時間)只更新一次。

  下面再討論一種客戶端的緩存技術這種技術也叫斷開連接的ADO連接技術(翻譯水平太次,聽上去怎麼這麼彆扭)。這種技術主要使用在用來保存用戶個人信息,例如用戶的密碼,代號等等上面。它主要使用了ADO的一些屬性。同時也回答了一些網友曾經提到過的能否在Application中使用ADO對象的問題。解釋不清楚,下面讓代碼來發言:

 

文件GLOBAL.ASA:


<!--METADATA TYPE="TypeLib" FILE="C:\Program Files\Common   
Files\system\ado\msado15.dll"-- >    
< SCRIPT LANGUAGE=VBScript RUNAT="Server" >    
Sub Application_OnStart    
SQL = "Select UserName, Password FROM UserInfo"    
cnnUsers = "DSN=User"    
Set rsUsers = Server.CreateObject("ADODB.Recordset")    
'注意下面這兩句話,就是用來實現那個叫可用的斷開連接的ADO技術    
rsCustomers.CursorLocation = adUseClient    
rsCustomers.Open SQL, cnnAdvWorks, adOpenStatic, AdLockReadOnly    
' 斷開RecordSet的和數據庫的連接    
rsCustomers.ActiveConnection = Nothing    
Set Application("rsCustomers") = rsCustomers    
End Sub
 

文件Users.ASP


< %    
'Clone方法使得每個用戶擁有自己的一個RecordSet集合    
Set yourUsers = Application("rsUsers").Clone    
Set UserName = yourUsers("UserName")    
Set Password = yourUsers("Password")    
Do Until yourUsers.EOF    
% >    
 
 用戶姓名:< %= UserName % > 用戶密碼:< %= Password % > 
 
 < %    
yourUsers.MoveNext    
Loop    
% >
 

於緩存的作用,我想我也不用再多說了,它的作用已經很明顯,特別是對於信息量非常大或是全數據庫頁面的網站,他能很好地利用主機的內存資源,加速ASP的執行效率,減輕服務器的負擔,而動網在這一方面做得是最突出的,像他現在的dvbbs7.1.0版,更是在緩存的利用上更上一層樓,前後臺大多的操作都和緩存有關,而現在動網裏用的也就是迷城浪子的緩存類:


< %    
Class Cls_Cache    
'==================使用說明====================    
'本類模塊是動網先鋒原創,作者:迷城浪子。如採用本類模塊,請不要去掉這個說明。
 '公有變量:Reloadtime 過期時間(單位爲分鐘)缺省值爲14400    
'MaxCount 緩存對象的最大值,超過則自動刪除使用次數少的對象。缺省值爲300    
'CacheName 緩存組的總名稱,缺省值爲"Dvbbs",如果一個站點中有超過一個緩存組,則需要外部改變這個值。    
'屬性:Name 定義緩存對象名稱,只寫屬性。    
'屬性:value 讀取和寫入緩存數據。    
'函數:ObjIsEmpty()判斷當前緩存是否過期。    
' 方法:DelCahe(MyCaheName)手工刪除一個緩存對象,參數是緩存對象的名稱。    
'========================    
Public Reloadtime,MaxCount,CacheName    
Private LocalCacheName,CacheData,DelCount    
 
 Private Sub Class_Initialize()    
Reloadtime=14400 '過期時間(單位爲分鐘)    
CacheName="Dvbbs" '緩存組的總名稱    
End Sub    
 
 Private Sub SetCache(SetName,NewValue)    
Application.Lock    
Application(SetName) = NewValue    
Application.unLock    
End Sub    
 
 Public Property Let Name(ByVal vNewValue) '緩存對象名稱    
LocalCacheName=LCase(vNewValue)    
End Property    
 
 Public Property Let Value(ByVal vNewValue) '讀取和寫入緩存數據    
If LocalCacheName<>"" Then    
CacheData=Application(CacheName"_"&LocalCacheName)    
If IsArray(CacheData) Then    
CacheData(0)=vNewValue    
CacheData(1)=Now()    
Else    
ReDim CacheData(2)    
CacheData(0)=vNewValue    
CacheData(1)=Now()    
End If    
SetCache CacheName"_"&LocalCacheName,CacheData    
Else    
Err.Raise vbObjectError + 1, "DvbbsCacheServer", " please change the CacheName."    
End If    
End Property    
 
 Public Property Get Value()    
If LocalCacheName<>"" Then    
CacheData=Application(CacheName"_"&LocalCacheName)    
If IsArray(CacheData) Then    
Value=CacheData(0)    
Else    
Err.Raise vbObjectError + 1, "DvbbsCacheServer", " The CacheData Is Empty."    
End If    
Else    
Err.Raise vbObjectError + 1, "DvbbsCacheServer", " please change the CacheName."    
End If    
End Property    
 
 Public Function ObjIsEmpty() '判斷當前緩存是否過期    
ObjIsEmpty=True    
CacheData=Application(CacheName"_"&LocalCacheName)    
If Not IsArray(CacheData) Then Exit Function    
If Not IsDate(CacheData(1)) Then Exit Function    
If DateDiff("s",CDate(CacheData(1)),Now()) < 60*Reloadtime Then ObjIsEmpty=False    
End Function    
 
 Private Sub makeEmpty(SetName) '釋放內存    
Application.Lock    
Application(SetName) = Empty    
Application.unLock    
End Sub    
 
 Public Sub DelCahe(MyCaheName) '刪除緩存    
makeEmpty(CacheName"_"&MyCaheName)    
End Sub    
End Class    
 
 'Set WydCache=New Cls_Cache    
'WydCache.Reloadtime=0.5 '定義過期時間 (以分鐘爲單位)    
'WydCache.CacheName="pages" '定義緩存名    
'IF WydCache.ObjIsEmpty() Then ''判斷是否可用(包括過期,與是否爲空值)    
'Response.write WydCache.Value    
'Else    
'..................    
'BoardJumpList=xxx    
'WydCache.Value=BoardJumpList '寫入內容    
'Response.write BoardJumpList    
'End if    
'    
'mycache.DelCahe(" 緩存名") 刪除緩存    
%>
 

以上內容轉自:https://www.jb51.net/article/52855.htm 


緩存(cache)是將相對高速的存儲設備作爲相對低速的存儲設備與系統之間I/O的緩衝區,它能大幅提高系統的性能.
  對於Web站點來說,緩存數據就是將用戶每次訪問你的站點時需要動態生成的信息預先生成並存儲在內存中,以靜態的形式送給用戶. 舉個例子吧:假設你的站點有一個更新不是很頻繁的下拉式列表框,列表框中的選項當用戶訪問站點時從數據庫中取出,我們可以將這些選項預先取出,存放在內存中以提高性能.

Application對象
  Application 對象能保存整個應用的全局信息(session對象保存某個特定用戶的信息). 每個站點都被認爲是一個應用,你可以在global.asa中利用Application Object存儲你的信息.
  讓我們來看看示例程序. 在這個例子裏,我將Application對象作爲一個變量用來存儲下拉菜單的菜單項. 當每次需要從數據庫中讀取菜單項的時候,我們從變量中讀取,避免了對數據庫的頻繁訪問.

請看以下的代碼:

<%= Application("ListBox")%>

<%
'把application 對象定義成一個變量
ListBox = Application("ListBox")

'檢查存儲變量的設置
If ListBox = "" Then

'如果存儲變量未設置,就直從數據庫中提取記錄項

Set oConn = Server.CreateObject("ADODB.Connection")
oConn.Open "DSN=travel;UID=;PWD="

sql = "SELECT * FROM types"

Set rs = oConn.Execute(sql)

crlf = chr(13) & chr(10)

'把數據庫中的數據加入 'ListBox'
'variable using a loop
ListBox = "<select name='listbox'>" & crlf
Do Until rs.EOF

Listbox = ListBox & " <option>" & _
rs("type") & "</option>" & crlf
rs.MoveNext

Loop

'把生成的Listbox存儲在Application對象中
Application("ListBox") = ListBox

End If
%>

  從代碼中我們可以看出,數據只有第一次被讀取時訪問數據庫,然後將之存儲在Application對象中,以後每次都從緩存中讀取,這樣將會加快站點的速度. 並不是每個站點都需要緩存來提高速度,但如果你的站點上運行着複雜的應用程序,且你想讓系統獲得最優的性能,那麼你最好考慮一下緩存數據.

以上內容轉自:https://blog.csdn.net/RoomDay/article/details/369246
————————————————

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章