八、設(shè)置HTTP應(yīng)答頭作 者 : 仙人掌工作室
8.1 HTTP應(yīng)答頭概述
Web服務(wù)器的HTTP應(yīng)答一般由以下幾項(xiàng)構(gòu)成:一個(gè)狀態(tài)行,一個(gè)或多個(gè)應(yīng)答頭,一個(gè)空行,內(nèi)容文檔。設(shè)置HTTP應(yīng)答頭往往和設(shè)置狀態(tài)行中的狀態(tài)代碼結(jié)合起來(lái)。例如,有好幾個(gè)表示“文檔位置已經(jīng)改變”的狀態(tài)代碼都伴隨著一個(gè)Location頭,而401(Unauthorized)狀態(tài)代碼則必須伴隨一個(gè)WWW-Authenticate頭。
然而,即使在沒(méi)有設(shè)置特殊含義的狀態(tài)代碼時(shí),指定應(yīng)答頭也是很有用的。應(yīng)答頭可以用來(lái)完成:設(shè)置Cookie,指定修改日期,指示瀏覽器按照指定的間隔刷新頁(yè)面,聲明文檔的長(zhǎng)度以便利用持久HTTP連接,……等等許多其他任務(wù)。
設(shè)置應(yīng)答頭最常用的方法是HttpServletResponse的setHeader,該方法有兩個(gè)參數(shù),分別表示應(yīng)答頭的名字和值。和設(shè)置狀態(tài)代碼相似,設(shè)置應(yīng)答頭應(yīng)該在發(fā)送任何文檔內(nèi)容之前進(jìn)行。
setDateHeader方法和setIntHeadr方法專門用來(lái)設(shè)置包含日期和整數(shù)值的應(yīng)答頭,前者避免了把Java時(shí)間轉(zhuǎn)換為GMT時(shí)間字符串的麻煩,后者則避免了把整數(shù)轉(zhuǎn)換為字符串的麻煩。
HttpServletResponse還提供了許多設(shè)置常見(jiàn)應(yīng)答頭的簡(jiǎn)便方法,如下所示:
setContentType:設(shè)置Content-Type頭。大多數(shù)Servlet都要用到這個(gè)方法。 setContentLength:設(shè)置Content-Length頭。對(duì)于支持持久HTTP連接的瀏覽器來(lái)說(shuō),這個(gè)函數(shù)是很有用的。 addCookie:設(shè)置一個(gè)Cookie(Servlet API中沒(méi)有setCookie方法,因?yàn)閼?yīng)答往往包含多個(gè)Set-Cookie頭)。 另外,如上節(jié)介紹,sendRedirect方法設(shè)置狀態(tài)代碼302時(shí)也會(huì)設(shè)置Location頭。 8.2 常見(jiàn)應(yīng)答頭及其含義
有關(guān)HTTP頭詳細(xì)和完整的說(shuō)明,請(qǐng)參見(jiàn)http://www.w3.org/Protocols/規(guī)范。
應(yīng)答頭 說(shuō)明 Allow 服務(wù)器支持哪些請(qǐng)求方法(如GET、POST等)。 Content-Encoding 文檔的編碼(Encode)方法。只有在解碼之后才可以得到Content-Type頭指定的內(nèi)容類型。利用gzip壓縮文檔能夠顯著地減少HTML文檔的下載時(shí)間。Java的GZIPOutputStream可以很方便地進(jìn)行g(shù)zip壓縮,但只有Unix上的Netscape和Windows上的IE 4、IE 5才支持它。因此,Servlet應(yīng)該通過(guò)查看Accept-Encoding頭(即request.getHeader("Accept-Encoding"))檢查瀏覽器是否支持gzip,為支持gzip的瀏覽器返回經(jīng)gzip壓縮的HTML頁(yè)面,為其他瀏覽器返回普通頁(yè)面。 Content-Length 表示內(nèi)容長(zhǎng)度。只有當(dāng)瀏覽器使用持久HTTP連接時(shí)才需要這個(gè)數(shù)據(jù)。如果你想要利用持久連接的優(yōu)勢(shì),可以把輸出文檔寫(xiě)入ByteArrayOutputStram,完成后查看其大小,然后把該值放入Content-Length頭,通過(guò)byteArrayStream.writeTo(response.getOutputStream()發(fā)送內(nèi)容。 Content-Type 表示后面的文檔屬于什么MIME類型。Servlet默認(rèn)為text/plain,但通常需要顯式地指定為text/html。由于經(jīng)常要設(shè)置Content-Type,因此HttpServletResponse提供了一個(gè)專用的方法setContentTyep。 Date 當(dāng)前的GMT時(shí)間。你可以用setDateHeader來(lái)設(shè)置這個(gè)頭以避免轉(zhuǎn)換時(shí)間格式的麻煩。 Expires 應(yīng)該在什么時(shí)候認(rèn)為文檔已經(jīng)過(guò)期,從而不再緩存它? Last-Modified 文檔的改動(dòng)時(shí)間?蛻艨梢酝ㄟ^(guò)If-Modified-Since請(qǐng)求頭提供一個(gè)日期,該請(qǐng)求將被視為一個(gè)條件GET,只有改動(dòng)時(shí)間遲于指定時(shí)間的文檔才會(huì)返回,否則返回一個(gè)304(Not Modified)狀態(tài)。Last-Modified也可用setDateHeader方法來(lái)設(shè)置。 Location 表示客戶應(yīng)當(dāng)?shù)侥睦锶ヌ崛∥臋n。Location通常不是直接設(shè)置的,而是通過(guò)HttpServletResponse的sendRedirect方法,該方法同時(shí)設(shè)置狀態(tài)代碼為302。 Refresh 表示瀏覽器應(yīng)該在多少時(shí)間之后刷新文檔,以秒計(jì)。除了刷新當(dāng)前文檔之外,你還可以通過(guò)setHeader("Refresh", "5; URL=http://host/path")讓瀏覽器讀取指定的頁(yè)面。 注意這種功能通常是通過(guò)設(shè)置HTML頁(yè)面HEAD區(qū)的<META HTTP-EQUIV="Refresh" CONTENT="5;URL=http://host/path">實(shí)現(xiàn),這是因?yàn),自?dòng)刷新或重定向?qū)τ谀切┎荒苁褂肅GI或Servlet的HTML編寫(xiě)者十分重要。但是,對(duì)于Servlet來(lái)說(shuō),直接設(shè)置Refresh頭更加方便。
注意Refresh的意義是“N秒之后刷新本頁(yè)面或訪問(wèn)指定頁(yè)面”,而不是“每隔N秒刷新本頁(yè)面或訪問(wèn)指定頁(yè)面”。因此,連續(xù)刷新要求每次都發(fā)送一個(gè)Refresh頭,而發(fā)送4狀態(tài)代碼則可以阻止瀏覽器繼續(xù)刷新,不管是使用Refresh頭還是<META HTTP-EQUIV="Refresh" ...>。
注意Refresh頭不屬于HTTP 1.1正式規(guī)范的一部分,而是一個(gè)擴(kuò)展,但Netscape和IE都支持它。
Server 服務(wù)器名字。Servlet一般不設(shè)置這個(gè)值,而是由Web服務(wù)器自己設(shè)置。 Set-Cookie 設(shè)置和頁(yè)面關(guān)聯(lián)的Cookie。Servlet不應(yīng)使用response.setHeader("Set-Cookie", ...),而是應(yīng)使用HttpServletResponse提供的專用方法addCookie。參見(jiàn)下文有關(guān)Cookie設(shè)置的討論。 WWW-Authenticate 客戶應(yīng)該在Authorization頭中提供什么類型的授權(quán)信息?在包含401(Unauthorized)狀態(tài)行的應(yīng)答中這個(gè)頭是必需的。例如,response.setHeader("WWW-Authenticate", "BASIC realm=\"executives\"")。 注意Servlet一般不進(jìn)行這方面的處理,而是讓W(xué)eb服務(wù)器的專門機(jī)制來(lái)控制受密碼保護(hù)頁(yè)面的訪問(wèn)(例如.htaccess)。
8.3 實(shí)例:內(nèi)容改變時(shí)自動(dòng)刷新頁(yè)面
下面這個(gè)Servlet用來(lái)計(jì)算大素?cái)?shù)。因?yàn)橛?jì)算非常大的數(shù)字(例如500位)可能要花不少時(shí)間,所以Servlet將立即返回已經(jīng)找到的結(jié)果,同時(shí)在后臺(tái)繼續(xù)計(jì)算。后臺(tái)計(jì)算使用一個(gè)優(yōu)先級(jí)較低的線程以避免過(guò)多地影響Web服務(wù)器的性能。如果計(jì)算還沒(méi)有完成,Servlet通過(guò)發(fā)送Refresh頭指示瀏覽器在幾秒之后繼續(xù)請(qǐng)求新的內(nèi)容。
注意,本例除了說(shuō)明HTTP應(yīng)答頭的用處之外,還顯示了Servlet的另外兩個(gè)很有價(jià)值的功能。首先,它表明Servlet能夠處理多個(gè)并發(fā)的連接,每個(gè)都有自己的線程。Servlet維護(hù)了一份已有素?cái)?shù)計(jì)算請(qǐng)求的Vector表,通過(guò)查找素?cái)?shù)個(gè)數(shù)(素?cái)?shù)列表的長(zhǎng)度)和數(shù)字個(gè)數(shù)(每個(gè)素?cái)?shù)的長(zhǎng)度)將當(dāng)前請(qǐng)求和已有請(qǐng)求相匹配,把所有這些請(qǐng)求同步到這個(gè)列表上。第二,本例證明,在Servlet中維持請(qǐng)求之間的狀態(tài)信息是非常容易的。維持狀態(tài)信息在傳統(tǒng)的CGI編程中是一件很麻煩的事情。由于維持了狀態(tài)信息,瀏覽器能夠在刷新頁(yè)面時(shí)訪問(wèn)到正在進(jìn)行的計(jì)算過(guò)程,同時(shí)也使得Servlet能夠保存一個(gè)有關(guān)最近請(qǐng)求結(jié)果的列表,當(dāng)一個(gè)新的請(qǐng)求指定了和最近請(qǐng)求相同的參數(shù)時(shí)可以立即返回結(jié)果。
PrimeNumbers.java
注意,該Servlet要用到前面給出的ServletUtilities.java。另外還要用到:PrimeList.java,用于在后臺(tái)線程中創(chuàng)建一個(gè)素?cái)?shù)的Vector;Primes.java,用于隨機(jī)生成BigInteger類型的大數(shù)字,檢查它們是否是素?cái)?shù)。(此處略去PrimeList.java和Primes.java的代碼。) package hall;
import java.io.*;import javax.servlet.*;import javax.servlet.http.*;import java.util.*;
public class PrimeNumbers extends HttpServlet {private static Vector primeListVector = new Vector();private static int maxPrimeLists = 30;public void doGet(HttpServletRequest request,HttpServletResponse response)throws ServletException, IOException {int numPrimes = ServletUtilities.getIntParameter(request, "numPrimes", 50);int numDigits = ServletUtilities.getIntParameter(request, "numDigits", 1);PrimeList primeList = findPrimeList(primeListVector, numPrimes, numDigits);if (primeList == null) {primeList = new PrimeList(numPrimes, numDigits, true);synchronized(primeListVector) {if (primeListVector.size() >= maxPrimeLists)primeListVector.removeElementAt(0);primeListVector.addElement(primeList);}}Vector currentPrimes = primeList.getPrimes();int numCurrentPrimes = currentPrimes.size();int numPrimesRemaining = (numPrimes - numCurrentPrimes);boolean isLastResult = (numPrimesRemaining == 0);if (!isLastResult) {response.setHeader("Refresh", "5");}response.setContentType("text/html");PrintWriter out = response.getWriter();String title = "Some " + numDigits + "-Digit Prime Numbers";out.println(ServletUtilities.headWithTitle(title) +"<BODY BGCOLOR=\"#FDF5E6\">\n" +"<H2 ALIGN=CENTER>" + title + "</H2>\n" +"<H3>Primes found with " + numDigits +" or more digits: " + numCurrentPrimes + ".</H3>");if (isLastResult)out.println("<B>Done searching.</B>");elseout.println("<B>Still looking for " + numPrimesRemaining +" more<BLINK>...</BLINK></B>");out.println("<OL>");for(int i=0; i<numCurrentPrimes; i++) {out.println(" <LI>" + currentPrimes.elementAt(i));}out.println("</OL>");out.println("</BODY></HTML>");}
public void doPost(HttpServletRequest request,HttpServletResponse response)throws ServletException, IOException {doGet(request, response);}
// 檢查是否存在同類型請(qǐng)求(已經(jīng)完成,或者正在計(jì)算)。// 如存在,則返回現(xiàn)有結(jié)果而不是啟動(dòng)新的后臺(tái)線程。private PrimeList findPrimeList(Vector primeListVector,int numPrimes,int numDigits) {synchronized(primeListVector) {for(int i=0; i<primeListVector.size(); i++) {PrimeList primes = (PrimeList)primeListVector.elementAt(i);if ((numPrimes == primes.numPrimes()) &&(numDigits == primes.numDigits()))return(primes);}return(null);}}}
PrimeNumbers.html <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"><HTML><HEAD><TITLE>大素?cái)?shù)計(jì)算</TITLE></HEAD><CENTER><BODY BGCOLOR="#FDF5E6"><FORM ACTION="/servlet/hall.PrimeNumbers"><B>要計(jì)算幾個(gè)素?cái)?shù):</B><INPUT TYPE="TEXT" NAME="numPrimes" VALUE=25 SIZE=4><BR><B>每個(gè)素?cái)?shù)的位數(shù):</B><INPUT TYPE="TEXT" NAME="numDigits" VALUE=150 SIZE=3><BR><INPUT TYPE="SUBMIT" VALUE="開(kāi)始計(jì)算"></FORM></CENTER></BODY></HTML>
來(lái)源:http://edu.chinaz.com