直播中
1. 緩存和“到期”ASP網(wǎng)頁
用戶的瀏覽器以及他們和服務(wù)器這間的任一代理服務(wù)器,都可以緩存HTML和用ASP創(chuàng)建的網(wǎng)頁。當(dāng)用戶隨后請求頁面時(shí),瀏覽器就發(fā)送一個(gè)“最新修改”的請求到服務(wù)器(使用一個(gè)包含緩存版本的日期的HTTP_IF_MODIFIED_SINCE報(bào)頭),詢問網(wǎng)頁是否已被修改。
若沒有被修改,服務(wù)器應(yīng)用狀態(tài)碼和消息“304 Not Modified”來響應(yīng),瀏覽器將使用緩存的內(nèi)容而不會通過網(wǎng)絡(luò)下載一個(gè)副本。若已經(jīng)存在已修改的版本,它就會與“200 OK”狀態(tài)碼和消息一道被發(fā)送出去。
1) Response.CacheContol屬性
其他的一些因素也會影響這一處理過程。然而,任一被網(wǎng)頁使用的網(wǎng)絡(luò)路由內(nèi)的代理服務(wù)器(一般位于客戶機(jī)端),能被通過設(shè)置Response.CacheControl屬性為Private來放棄緩存網(wǎng)頁。在ASP 3.0中對ASP網(wǎng)頁這是缺省的,不用設(shè)置。但在網(wǎng)頁為個(gè)別訪問者特別定制時(shí)尤其有用。這可以阻止別的在同一網(wǎng)絡(luò)上的用戶進(jìn)入同一網(wǎng)頁。當(dāng)CacheControl的屬性值被設(shè)定為Public時(shí),允許服務(wù)器緩存網(wǎng)頁。注意,一些代理服務(wù)器可能表現(xiàn)得不盡相同,或忽視或越過這個(gè)報(bào)頭。
在IE4中,在代理服務(wù)器緩存可用時(shí),有可能得到一個(gè)虛假的“This page has expired”消息。我們已提供了一個(gè)網(wǎng)頁(expiretest_form.asp),可以通過自己的代理服務(wù)器在網(wǎng)絡(luò)上做試驗(yàn),來檢查這一屬性的影響??梢酝ㄟ^在“Response Object”主頁中單擊“Response. CacheControl”鏈接來顯示這個(gè)網(wǎng)頁。如下圖所示:
這一頁面提交到expiretest_result.asp網(wǎng)頁時(shí),能夠設(shè)置Response.CacheControl屬性,然后在網(wǎng)頁中插入值和腳本被執(zhí)行的時(shí)間:
<%
If Request.Form(“public”) = “on” Then ‘Cache-Control check box was ticked
Response.CacheControl = “Public”
Else
Response.CacheControl = “Private”
End If
%>
<HTML>
...
Cache-Control is: <B><% = Response.CacheControl %></B><P>
Value in text box is: <B><% Response.Write Request.Form(“textbox”) %>
<%
Response.Write Right(“0” & Hour(Now),2) & “:” & Right(“0” & Minute(Now),_
& 2) & “:” & Right(“0” & Second(Now),2)
%></B>
通過單擊瀏覽器上的“Back”和“Forward”,能看到代碼是自動執(zhí)行還是使用緩存的副本,如下圖所示。結(jié)果隨瀏覽器的不同而變化。
2) Response.Expires和Response.ExpiresAbsolute屬性
控制緩存的網(wǎng)頁存放時(shí)間的兩個(gè)屬性為Response對象的Expires和ExpriesAbsolute屬性。Response.Expires定義了風(fēng)頁在從緩存區(qū)被丟棄前應(yīng)保持有效的時(shí)間長度,以創(chuàng)建以來的分鐘數(shù)形式表示。ExpiresAbsolute屬性為到期時(shí)間設(shè)置了一個(gè)絕對的日期和時(shí)間。
我們提供一個(gè)命名為addheaders_form.asp的例子網(wǎng)頁,用于演示如何使用這些屬性。在“Response Object”主頁中單擊對這兩種屬性的鏈接,如下圖所示:
在這一頁面中,可加入自己定制的HTTP報(bào)頭,并可設(shè)置一些影響響應(yīng)的HTTP報(bào)頭的多種屬性。在“提交查詢內(nèi)容”按鈕上單擊時(shí),頁面show_headers.asp在返回的數(shù)據(jù)流中添加所選的報(bào)頭,然后顯示用來完成此操作的代碼,顯示相應(yīng)的執(zhí)行時(shí)間,可用來檢查頁面是被緩存還是被再次執(zhí)行,如下圖所示:
show_headers.asp網(wǎng)頁中的代碼創(chuàng)建和添加HTTP報(bào)頭,程序如下:
<%
‘Write HTTP headers before any other output
If Request.Form(“expires”) = “on” Then _
Response.Expires = Request.Form(“expires_value”)
If Request.Form(“expiresabs”) = “on” Then _
Response.ExpiresAbsolute = Request.Form(“expiresabs_value”)
If Request.Form(“l(fā)astmod”) = “on” Then _
Response.AddHeader “LAST-MODIFIED”, Cstr(Request.Form(“l(fā)astmod_value”))
If Request.Form(“pragma”) = “on” Then _
Response.AddHeader “PRAGMA”, CStr(Request.Form(“pragma_value”))
If Request.Form(“refresh”) = “on” Then _
Response.AddHeader “REFRESH”, CStr(Request.Form(“refresh_value”))
If Request.Form(“addheader”) = “on” And Len(Request.Form(“addheader_name”)) Then _
Response.AddHeader CStr(Request.Form(“addheader_name”)), _
CStr(Request.Form(“addheader_value”))
If Request.Form(“status”) = “on” Then _
Response.Status = Request.Form(“status_value”)
%>
<HTML>
...
... Show code and execution time
...
其余部分僅僅是顯示已被執(zhí)行的代碼和執(zhí)行時(shí)間。讀者會注意到包含在網(wǎng)頁中的定制的報(bào)頭“PRAGMA”(至今我們還沒討論過)。一些(先前的)代理服務(wù)器使用它作為網(wǎng)磁是否應(yīng)被緩存的指示。缺省是網(wǎng)頁被緩沖,除非接受到HTTP報(bào)頭“PRAGMA=NO-CACHE“。
2. 創(chuàng)建狀態(tài)碼和定制的HTTP報(bào)頭
可使用先前在實(shí)例網(wǎng)頁中所看到的Response對象的AddHeader方法來創(chuàng)建自己的狀態(tài)碼或自己喜歡的定制的報(bào)頭。這一方法需要兩個(gè)參數(shù):HTTP報(bào)頭名稱或一個(gè)包含其值或分配給它的值的字符串。作為一個(gè)例子,下面的代碼在頁面中添加REFRESH報(bào)頭:
Response.AddHeader “REFRESH”, ”60;URL=newpath/newpage.asp”
這等同于客戶機(jī)端<META>元素:
<META HTTP-EQUIV=”REFRESH”, “60;URL=newpath/newpage.asp”
換句話說,也可配合Status屬性使用AddHeader方法使瀏覽器載入一個(gè)新的頁面:
Response.Status = “302 Object Moved”
Response.Addheader “Location”, “newpath/newpage.asp”
這等同于使用Response.Redirect方法:
Response.Redirect “newpath/newpage.asp”
Response.Status屬性可被用來發(fā)送一些所需要的狀態(tài)消息,例如添加如下幾行:
Response.Status= “401 Unauthorized”
Response.Addheader “WWW-Authenticate”, “BASIC”
強(qiáng)制瀏覽器顯示一個(gè)用戶名/口令對話框,然后使用BASIC驗(yàn)證把它們發(fā)送回服務(wù)器(將在本書后續(xù)部分看到驗(yàn)證方法)。
3. MIME類型和內(nèi)容類型
當(dāng)我們想向?yàn)g覽器發(fā)送一個(gè)動態(tài)創(chuàng)建的字符串,而且它們自己提供給瀏覽器時(shí)沒有直接指明內(nèi)容類型,而是提供表示是否是磁盤文件的擴(kuò)展名時(shí),Response.ContentType是非常有用的。除非特別指定,所有ASP創(chuàng)建的網(wǎng)頁缺省都為“text/type”。內(nèi)容類型的標(biāo)識符是MIME類型(MIME代表Multi-purpose Internet Multimedia Extension或Multi-pupose Internet Mail Extension,通常依據(jù)上下文來定)。
例如,若發(fā)送到客戶的數(shù)據(jù)注解是通過從數(shù)據(jù)庫讀二進(jìn)制值創(chuàng)建的圖片,就需要在發(fā)送任何內(nèi)容之前添加合適的CONTENT-TYPE報(bào)頭:
Response.ContentType = “image/jpeg”
假如從一個(gè)數(shù)據(jù)庫創(chuàng)建一個(gè)XML文件,使用MIEM類型“text/xml”;并且如果正在創(chuàng)建一個(gè)文本文件可以在文件編輯器中顯示或作為一個(gè)磁盤文件在客戶上被存儲起來,使用“text/text”。
4. 添加PICS卷標(biāo)
Respnse.Pics屬性僅僅是添加一個(gè)PICS(Platform for Internet Content system)卷標(biāo)到頁面上,方式與通常用<META>標(biāo)記所用的方式相同:
QUOT = Chr(34)
StrPicsLabel = “(PICS-1.0” & QUOT & “http://www.rsac.org/ratingsv01.html”_
& QUOT & “ 1 gen true comment “ & QUOT _
& “RSACi North America Server” & QUOT & “ for “ & QUOT _
& “http://yoursite.com” & QUOT & “ on “ & QUOT _
& “1999.08.01T03:04-0500” & QUOT & “ r (n 0 s 0 v 2 l 3))”
Response.Pics(strPicsLabel)
這段代碼添加了如下的PICS卷標(biāo):
(PICS-1.0 “http://www.rsac.org/ratingsv01.html” 1 gen true comment “RSACi
North America Server” for “http://yoursite.com” on “1999.08.01T03:04-0500”
r (n 0 s 0 v 2 l 3))
要得到關(guān)于PICS的更多的信息,或了解更多的定義頁面內(nèi)容的方式,請檢索http://www.rsac.org/站點(diǎn)。
在Internet Service Manager中定義報(bào)頭
在第1章,已經(jīng)說明了如何在Internet Service Manage(MMC插件)應(yīng)用程序中設(shè)置每個(gè)Web網(wǎng)站和IIS 5.0目錄的屬性,這就定義了使用此站點(diǎn)或目錄資源發(fā)送到客戶機(jī)的所有請求的HTTP報(bào)頭,也就提供了使用每個(gè)網(wǎng)頁中的ASP腳本代碼設(shè)置這些屬性的替代方法。
在Web站點(diǎn)或目錄上右擊鼠標(biāo)并選擇“Properties”,在其對話框的“HTTP Headers”選項(xiàng)卡中,可設(shè)置頁面內(nèi)容有效期的相對時(shí)間或絕對日期,定義定制的報(bào)頭,創(chuàng)建PICS內(nèi)容等級標(biāo)簽,也可以通過MIME類型映射來定義內(nèi)容類型,如下圖所示:
在上圖中,可以看到已創(chuàng)建了自定義的REFRESH HTTP報(bào)頭,應(yīng)用于從此目錄載入的所有網(wǎng)頁。即每一分鐘自動地重載(刷新)一次(對于顯示棒球比賽的最近比分是非常理想的,但對服務(wù)器而言負(fù)擔(dān)太重了)。Custom HTTP Headers欄的Edit對話框如下圖所示:
要在“MIME Map”框中添加自定義的內(nèi)容類型映射,只需在“Properties”主對話框中單擊“File Types”按扭把它們添加到清單中即可,如下圖所示:
當(dāng)使用HTTP報(bào)頭開始試驗(yàn)時(shí),你很快會發(fā)現(xiàn)不是所有的瀏覽器表現(xiàn)都相同,許多瀏覽器以不同的方式響應(yīng)不同的HTTP報(bào)頭,使得可靠地建立一個(gè)普遍適用的原則有時(shí)極為困難。
2. 使用客戶證書
假如設(shè)立了一個(gè)安全的Web網(wǎng)站或部分內(nèi)容具有安全機(jī)制的網(wǎng)站,可安裝一個(gè)數(shù)字服務(wù)器證書,通過允許訪問者使用證書中的加密的細(xì)節(jié),來驗(yàn)證服務(wù)器。每一次對該站點(diǎn)或目錄的頁面請求,服務(wù)器都將發(fā)送證書的一個(gè)副本,瀏覽器可檢查這個(gè)副本以確定正在和誰交談。
同樣,也可設(shè)置服務(wù)器,要求用戶在進(jìn)入網(wǎng)站時(shí)提供一個(gè)有效的數(shù)字證書。他們可從很多來源獲得此證書,例如Verisign(http://www.verisign.com)或Thawte Consulting(http://www.thawte.com)。讀者將在第25章看到這一處理過程的細(xì)節(jié)。
這些情況都使用了Request對象的ClientCertificate集合的值,本章的實(shí)例代碼中,已包含了一個(gè)顯示用戶如何使用些集合值的一些方法的頁面。
這一網(wǎng)頁被命名為showcert.asp,而且其所做的一切就是遍歷ClientCertificate集合顯示其包含的所有值??墒褂靡郧敖?jīng)常使用的簡單代碼來完成它,唯一的不同之處就是建立一個(gè)HTML表以容納結(jié)果,并將其截為每60個(gè)字符一組。
<TABEL CELLPADDING=0 CELLSPACING=0>
<%
For Each keyItem In Request.ClientCertificate()
StrItemValue = Request.ClientCertificate(keyItem)
If Len(strItemValue) > 90 Then strItemValue = Left(strItemValue, 60) & “..etc.”
Response.Write “<TR><TD>” & keyItem & “ = “ & strItemValue & “</TD></TR>”
Next
%>
</TABLE>
運(yùn)行結(jié)果如下圖所示:(由于豆豆沒有申請服務(wù)器證書,該圖略)
使用客戶證書重定向
一旦要求所有訪問網(wǎng)站或部分網(wǎng)站的瀏覽者給出的其客戶證書,就可以使用其包含的信息來制作我們?yōu)榇擞脩魟?chuàng)建的網(wǎng)頁。例如,可使用他們的證書的Organization條目來自動使他們重定向到該網(wǎng)站的指定部分,使別的訪問者重定向到別的地方:
If Request.ClientCertificate(“SubjectO”) = “Wrox Press Inc” Then
Response.Redirect “/wrox_staff/default.asp” ‘Wrox staff site
Else
Response.Redirect “/public/Default.asp” ‘Normal public site
End If
相應(yīng)地,可使用Country條目來使訪問者重定向到一個(gè)相應(yīng)的網(wǎng)站:
Select Case Request.ClientCertificate(“SubjectC”)
Case “UK”: Response.Redirect “http://uk_site.co.uk/”
Case “DE”: Response.Redirect “http://de_site.co.de/”
Case “FR”: Response.Redirect “http://fr_site.co.fr/”
‘... ect.
Case Else: Response.Redirect “http://us_site.com/”
End Select
3. 讀寫二進(jìn)制數(shù)據(jù)
有兩個(gè)方法提供了對從瀏覽器發(fā)送到服務(wù)器的HTTP數(shù)據(jù)流和從服務(wù)器返回到瀏覽器的數(shù)據(jù)流的二進(jìn)制數(shù)據(jù)訪問。Request.BinaryRead方法可得到指定要讀取的字節(jié)數(shù)的參數(shù),并返回變體類型的數(shù)組,其中包含從請求的POST段中得到的字節(jié)(例如在ASP的Form集合中數(shù)據(jù))。下面的程序讀數(shù)據(jù)的頭64個(gè)字節(jié):
varContent = Request.BinaryRead(64)
假如使用了BinaryRead方法,以后就不能訪問ASP的Request.Form集合。同樣,一旦我們采用任何方式引用了Request.Form集合,就不能使用BinaryRead方法。
把二進(jìn)制數(shù)據(jù)寫進(jìn)ASP創(chuàng)建的響應(yīng)流中也是可能的,可采用BinaryWrite方法。需要給其提供想寫到客戶的字節(jié)的變體類型數(shù)組:
Response.BinaryWrite(varContent)
這些方法都很少使用,除非從一個(gè)數(shù)據(jù)庫創(chuàng)建非HTML源才用到這些方法。使用的一個(gè)實(shí)例就是從數(shù)據(jù)庫讀取組成圖像的字節(jié),并使用BinaryWrite方法把它發(fā)送到客戶。
4. 創(chuàng)建定制的日志消息
假如設(shè)置了服務(wù)器,以W3C Extended Log File Format格式將請求記錄到一個(gè)文本文件,可使用Response.AppendToLog方法在日志文件條目的結(jié)尾處添加一條消息字符串。若想為特定的網(wǎng)頁存儲一些值或消息,或在腳本中出現(xiàn)了特定的情況時(shí),這種方式是非常有用的。
例如,通過的Intranet的“stationary order”應(yīng)用程序,可以記錄超過特定的條目數(shù)目的雇員的部門號碼:
...
If intItemCount > 25 Then
Response.AppendToLog “Large order from ‘” & strDept & department.”
End If
...
設(shè)置擴(kuò)展的日志
要使用AppendToLog方法,必須激活W3C Extended Log File Format日志設(shè)置。該設(shè)置方法是,進(jìn)入Properties對話框中的Web Site選項(xiàng)卡,選中Enable Logging復(fù)選框,選擇W3C Extended Log File Format并單擊Properties按鈕,如下圖所示:
在出現(xiàn)的Extended Logging Properties對話框中,可選擇想包括進(jìn)日志文件的條目。確保選中URI Stem,否則AppendToLog方法將失敗,如下圖所示:
我們提供了一個(gè)試圖在日志文件中寫入一個(gè)條目的簡單實(shí)例頁面,可從Request Object主頁(show_request.asp)中的AppendToLog方法鏈接處打開它。這一頁面所做的全部工作就是創(chuàng)建一個(gè)包含當(dāng)前日期和時(shí)間的簡單字符串,然后執(zhí)行AppendToLog方法:
strToAppend = “Page executed on ” & Now
Response.AppendToLog strToAppend
結(jié)果如下圖所示:
小結(jié)
本章已經(jīng)開始了對ASP 3.0的研究,而且我們也看到了ASP 3.0如何與Internet Informateion Server 5.0共同工作,以提供一個(gè)易用的、高效的創(chuàng)建動態(tài)Web網(wǎng)頁和Web應(yīng)用程序的方法。當(dāng)然,仍有一些地方需要去研究,本章僅僅是學(xué)習(xí)了ASP內(nèi)置的兩個(gè)最基本的對象。
這兩個(gè)最基本的對象是Request和Response對象,允許我們訪問和使用作為客戶機(jī)/服務(wù)器會話一部分的值,無論用戶何時(shí)從Web網(wǎng)站請求和載入一個(gè)網(wǎng)頁或資源,這種會話就會進(jìn)行,意味著Request對象能夠提供對用戶請求的全部內(nèi)容的訪問,同時(shí)Response對象允許創(chuàng)建和修改服務(wù)器發(fā)回的響應(yīng)。
這些對象能夠通過集合和屬性揭示會話的各個(gè)部分,并提供了多個(gè)能用來檢索和修改各段的方法。假如把它們當(dāng)作分解用戶請求和使用相應(yīng)的內(nèi)容創(chuàng)建響應(yīng)的工具,這有助你理解究竟發(fā)生了什么。這也將有助于理解各種方法如何影響客戶、服務(wù)器和正在創(chuàng)建的網(wǎng)頁。