国产精品色无码视频,国产av毛片影院精品资源,亚洲人成网站77777·c0m,囯产av无码片毛片一级,夜夜操www99视频,美女白嫩胸交在线观看,亚洲a毛片性生活

薈聚奇文、博采眾長、見賢思齊
當(dāng)前位置:公文素材庫 > 公文素材 > 范文素材 > JavaWeb工作原理

JavaWeb工作原理

網(wǎng)站:公文素材庫 | 時間:2019-05-28 14:32:00 | 移動端:JavaWeb工作原理

JavaWeb工作原理

[JavaWeb工作原理][第一章]一什么是JavaWeb?

JavaWeb是用Java技術(shù)來解決相關(guān)web互聯(lián)網(wǎng)領(lǐng)域的技術(shù)總和。web包括:web服務(wù)器和web客戶端兩部分,在第一個項目(貸款計息查詢服務(wù)器)的時候已經(jīng)知道在Web服務(wù)器的作用是接受客戶端請求,然后向客戶端返回一些結(jié)果.瀏覽器的作用是允許用戶請求服務(wù)器上的某個資源,并且向用戶顯示請求的結(jié)果,HTML用于告訴瀏覽器怎樣向用戶顯示內(nèi)容,HTTP是WEB上客戶端和服務(wù)器之間通信所用的協(xié)議

二HTTP協(xié)議

Http是一種超文本傳輸協(xié)議(HyperTextTransferProtocol),他是一套計算機(jī)在網(wǎng)中通信的一種規(guī)則,在TCP/IP體系結(jié)構(gòu)中,HTTP屬于應(yīng)用層協(xié)議,位于TCP/IP協(xié)議的頂層.HTTP是一種無狀態(tài)的協(xié)議,意思是指在WEB瀏覽器和WEB服務(wù)器之間不需要建立持久的連接,HTTP定義的事物處理由以下四步組成:

1.客戶端和WEB服務(wù)器建立連接

a)客戶端和服務(wù)器的連接就是與客戶端與服務(wù)器的一個TCPSocket套接字連接2.客戶端發(fā)送HTTP請求

a)請求包含:請求行--請求行是一個ACCII文本行,由請求的HTTP方法,請求的

URL,HTTP版本組成,中間用空格分開,

b)請求頭---HTTP協(xié)議使用HTTP頭來傳遞請求的元信息,

c)空行---發(fā)送回車符和退行,通知服務(wù)器以下不再有請求頭

d)消息體--HTTP請求中帶有查詢字符串時,如果是GET方法,查詢字符串或表單數(shù)據(jù)附加在請求行中,那么消息體就沒有內(nèi)容,如果是POST方法,查詢字符串或表單數(shù)據(jù)就添加在消息體重

3.服務(wù)器端接收客戶端的HTTP請求,生成HTTP相應(yīng)回發(fā)

a)響應(yīng)包含:狀態(tài)行---每個HTTP響應(yīng)以一個狀態(tài)行開頭,它由HTTP協(xié)議版本,

響應(yīng)狀態(tài)碼,響應(yīng)描述組成,中間用空格分開b)響應(yīng)頭---響應(yīng)頭與請求頭一樣,也是一個用冒號分隔符的名稱/值對,冒號前面

是HTTP頭得名稱,后面是HTTP頭得值c)空行---發(fā)送回車符和退行,通知服務(wù)器以下不再有響應(yīng)頭

d)消息體---要發(fā)送回客戶端的HTML文檔或其他要顯示的內(nèi)容等,WEB服務(wù)器

把要發(fā)送給客戶端的文檔信息放在消息體中

4.服務(wù)器端關(guān)閉連接,客戶端解析并回發(fā)響應(yīng),恢復(fù)頁面

a)HTTP響應(yīng)到達(dá)客戶端后,瀏覽器先解析HTTP響應(yīng)中的狀態(tài)行,查看請求是否

成功的狀態(tài)代碼,然后開始一步步解析響應(yīng)

三WEB服務(wù)器缺陷與CGI

WEB服務(wù)是被設(shè)計用來向客戶端提供HTTP服務(wù)的,它只是能向客戶端提供靜態(tài)網(wǎng)頁內(nèi)容.我們知道,靜態(tài)網(wǎng)頁只是原封不動的待在WEB服務(wù)器目錄中,服務(wù)器知道靜態(tài)頁面,并把它原樣傳回到客戶端,每個客戶端看到的頁面都是一樣的,所以WEB服務(wù)器它本身并不具備動態(tài)頁面,所以在最早有了解決辦法CGI,

CGI即通用網(wǎng)關(guān)接口(CommonGateWayInterface),是最早用于創(chuàng)建動態(tài)服務(wù)器端內(nèi)容的一種技術(shù),使用CGI,WEB服務(wù)器可以將客戶端的請求傳遞給一個外部程序,這個外部程序可以執(zhí)行,創(chuàng)建內(nèi)容,并且將響應(yīng)傳遞給客戶端,

但是他在處理一個請求的時候還行,但是在處理對個請求的時候是沒有一個用戶請求就新開一個進(jìn)程,我們都知道進(jìn)程是有獨立的內(nèi)存空間的,如果請求用戶多的時候那么就會導(dǎo)致服務(wù)器的崩潰,

四Java的解決方案

在java里,Servlet以及web容器,被設(shè)計用來解決CGI的問題,為WEB開發(fā)者創(chuàng)建一個健壯的服務(wù)器環(huán)境,

一、Servlet

a)Servlet是J2EE規(guī)范,

b)它是一個普通的JAVA類,集成于HttpServlet,用于處理動態(tài)頁面的響應(yīng)

c)它是與平臺無關(guān)的JAVA類,能夠編譯成平臺中立的字節(jié)碼,從而被基于JAVA技術(shù)的WEB服務(wù)器動態(tài)裝載和運行

d)Servlet在服務(wù)器端得工作主要是執(zhí)行如下任務(wù)

i.

讀取客戶端發(fā)送的顯示和隱式數(shù)據(jù)

ii.生成結(jié)果

iii.向客戶端發(fā)送顯示數(shù)據(jù)(文檔)和隱式數(shù)據(jù)(HTTP響應(yīng)數(shù)據(jù))

二、WEB容器

a)WEB容器主要是如Tomcat,Jboss,WebLogic,WebSphere,Oracle9iAS等,其中Tomcat是一種用的很廣泛的Web容器,它是一個開放源代碼的免費的中間件產(chǎn)品b)WEB容器必須對Servlet支持以下幾點

i.通信支持

ii.生命周期管理

iii.多線程支持iv.JSP支持v.處理安全性

三、Servlet與WEB容器配合處理請求和響應(yīng)

a)與GUI類似,Servlet允許一個請求被一個程序處理.并且使用同樣的程序產(chǎn)生動態(tài)的

響應(yīng),此外,Servlet特別定義了一個有效的生命周期,使得用單個進(jìn)程管理所有請求成為可能,它消除了CGI的多進(jìn)程缺陷,允許主進(jìn)程在多個Servlet和多個請求之間共享內(nèi)存資源.b)Servlet運行在一個主進(jìn)程或者一個父進(jìn)程中,每個用戶發(fā)送請求信息到服務(wù)器,就會在Servlet進(jìn)程里創(chuàng)建一個線程,它們共享一個資源c)最后,WEB容器和Servlet對請求和響應(yīng)的處理如下

i.客戶端向WEB服務(wù)器發(fā)起一個HTTP請求

ii.HTTP請求被WEB服務(wù)器幾首,如果請求的是靜態(tài)頁面,則由WEB服務(wù)器負(fù)責(zé)

處理,如果請求的是JAVAWEB組件(Servlet或者JSP),則移交給WEB容器,iii.WEB容器根據(jù)Servlet的配置文件(web.xml),確定調(diào)用的具體Servlet類,并把request對象,response對象傳給它

iv.Servlet通過request對象知道客戶端的使用者是誰,客戶的請求信息是什么和其

他的一些信息

v.一旦Servlet完成了請求的處理,WEB容器就會刷新response,把控制權(quán)返回給

WEB服務(wù)器

五JavaWeb應(yīng)用程序的組成

1.配置文件---每個WEB應(yīng)用程序包括一個配置文件,WEB.XML2.靜態(tài)文件和JSP

3.類文件和包---WEB應(yīng)用程序裝載和管理自定義的JAVA代碼

4.網(wǎng)頁可以放在WEB應(yīng)用程序的根目錄下,根據(jù)動態(tài)網(wǎng)頁或者靜態(tài)網(wǎng)頁放在不同的目

錄里

5.圖像一般會放在images子目錄中,不過這是習(xí)慣,不是必須的

6.Servlet類,JavaBean類---編譯為Class文件后是防在WEB-INF/classes目錄的7.Lib目錄用來包含應(yīng)用程序任何所需要的jar文件8.標(biāo)記描述其放在WEB-INF目錄下

9.Applet程序放在應(yīng)用的目錄下

10.WEB-INF目錄下存放web.xml部署描述文件器

剩下的WEB容器Tomcat主要是在實踐中應(yīng)用,在這里不做總結(jié)

擴(kuò)展閱讀:JavaWeb工作原理

淺析基于Java的Web服務(wù)器工作原理

一個Web服務(wù)器也被稱為HTTP服務(wù)器,它通過HTTP協(xié)議與客戶端通信。這個客戶端通常指的是Web瀏覽器。一個基于Java的Web服務(wù)器用到二個重要的類,java.net.Socket與java.net.ServerSocket,并通過HTTP消息通信。因此,本文從討論HTTP與這二個類開始,然后我將解釋一個與本文相關(guān)的簡單的Web應(yīng)用。

TheHypertextTransferProtocol(HTTP)

HTTP是一種讓W(xué)eb服務(wù)器與瀏覽器(客戶端)通過Internet發(fā)送與接收數(shù)據(jù)的協(xié)議。它是一個請求、響應(yīng)協(xié)議--客戶端發(fā)出一個請求,服務(wù)器響應(yīng)這個請求。HTTP運用可靠的TCP連接,通常用的TCP80端口。它的第一個版本是HTTP/0.9,然后被HTTP/1.0取代。當(dāng)前的版本是HTTP/1.1,由RFC2616(.pdf)定義。

本節(jié)主要對應(yīng)HTTP1.1,足夠使你充分理解由Web服務(wù)器程序發(fā)出的消息。如果你對更加詳細(xì)的知識有興趣,可以參考RFC2616。

在HTTP中,客戶端總是通過建立一個連接與發(fā)送一個HTTP請求來發(fā)起一個事務(wù)。服務(wù)器不能主動去與客戶端聯(lián)系,也不能給客戶端發(fā)出一個回叫連接?蛻舳伺c服務(wù)器端都可以提前中斷一個連接。例如,當(dāng)用一個瀏覽器下載一個文件時,你可以通過點擊“停止”鍵來中斷文件的下載,關(guān)閉與服務(wù)器的HTTP連接。

HTTP請求

一個HTTP請求包含三個部分:

Method-URI-Protocol/Version方法-地址-版本

Requestheader請求頭

Entitybody請求實體

下面是一個HTTP請求實例:

POST/servlet/default.jspHTTP/1.1

Accept:text/plain;text/htmlAccept-Language:en-gb

Connection:Keep-Alive

Host:localhost

Referer:

User-Agent:Mozilla/4.0(compatible;MSIE4.01;Windows98)

Content-Length:33

Content-Type:application/x-www-form-urlencoded

Accept-Encoding:gzip,deflate

LastName=Franks&FirstName=Michael

TheMethod-URI-Protocol/Version在這個請求的第一行:

POST/servlet/default.jspHTTP/1.1

其中POST是請求的類型。每個客戶端HTTP請求可以是HTTP規(guī)范中指定的許多請求類型中的一種。HTTP1.1支持七種類型的請求,它們是GET,POST,HEAD,OPTIONS,PUT,DELETE,TRACE。其中GET與POST是Internet應(yīng)用中經(jīng)常用到的二種請求類型。

URI完整地指定了Internet資源。一個URI通常被解析為相對服務(wù)器的根目錄。這樣,它應(yīng)該總是以一個"/"前綴開始。一個URL實際上是URI的一種類型。

Version指的是該HTTP請求所用到的HTTP協(xié)議版本。

請求頭包含了客戶端環(huán)境與請求實體的一些有用的信息。例如它包含瀏覽器設(shè)定的語言、實體的長度等等。每條請求頭用回車換行符(CRLF)分開。一個非常重要的空行分開了請求頭與實體,它標(biāo)志著實體內(nèi)容的開始。一些Internet開發(fā)書籍認(rèn)為這個CRLF空行是HTTP請求的第四個部分。

在上面的HTTP請求中,實體只是簡單以下的一行:

LastName=Franks&FirstName=Michael

在一個典型的HTTP請求中,請求實體內(nèi)容會長得多。

HTTP響應(yīng)

與請求相似,HTTP響應(yīng)也由三部分組成:

Protocol-Statuscode-Description協(xié)議狀態(tài)描述代碼

Responseheaders響應(yīng)頭

Entitybody響應(yīng)實體

以下是一個HTTP響應(yīng)的實例:

HTTP/1.1200OK

Server:Microsoft-IIS/4.0

Date:Mon,3Jan199813:13:33GMT

Content-Type:text/html

Last-Modified:Mon,11Jan199813:23:42GMT

Content-Length:1

WelcometoBrainySoftware

響應(yīng)頭的第一行類似請求頭的第一行,告訴你所用的協(xié)議是HTTP1.1,請求成功(200=success),以及沒有任何問題。

響應(yīng)頭類似請求頭也包含了一些有用的信息。響應(yīng)的實體響應(yīng)本身的HTML內(nèi)容。頭與實體之間由回車換行的空行(CRLF)分開。

Socket類

一個socket是一個網(wǎng)絡(luò)連接的端點,它使得一個應(yīng)用可以從網(wǎng)絡(luò)讀與寫。在不同電腦上的二個應(yīng)用軟件能夠通過收發(fā)字節(jié)流而彼此通信。要發(fā)一個信息到另一個應(yīng)用程序,你需要知道它的IP地址,以及它的socket端口號。在Java中,一個socket用java.net.Socket來實現(xiàn)。

要創(chuàng)建一個socket,你可以用Socket類中幾個構(gòu)建方法中的一個。其中一個接受主機(jī)名與端口號作為參數(shù):

newSocket("yahoo.com",80);

一旦你成功地創(chuàng)建了一個Socket類的實例,你就可以用它去發(fā)送與接收字節(jié)流了。要發(fā)送字節(jié)流,你需要呼叫Socket類的getOutputStream方法來得到一個java.io.OutputSteam對象。要發(fā)送文本到遠(yuǎn)程的程序,你通常需要從返回的OutputStream創(chuàng)建一個java.io.PrintWriter對象。要從連接的另一端接收字節(jié)流,你需要呼叫Socket類的getInputStream方法,它返回一個java.io.InputStream對象。

以下代碼創(chuàng)建一個可以與本地HTTP服務(wù)器通信的socket(127.0.0.1表示一個本地的主機(jī)),發(fā)送一個HTTP請求,并接收從服務(wù)器的響應(yīng)。它還創(chuàng)建一個StringBuffer對象來接受響應(yīng),并打印到控制臺。

Socketsocket=newSocket("127.0.0.1","8080");OutputStreamos=socket.getOutputStream();

booleanautoflush=true;

PrintWriterout=newPrintWriter(socket.getOutputStream(),

autoflush);

BufferedReaderin=newBufferedReader(

newInputStreamReader(socket.getInputStream()));

//sendanHTTPrequesttothewebserver

out.println("GET/index.jspHTTP/1.1");

out.println("Host:localhost:8080");

out.println("Connection:Close");

out.println();

//readtheresponse

booleanloop=true;

StringBuffersb=newStringBuffer(8096);

while(loop){

if(in.ready()){inti=0;

while(i!=-1){

i=in.read();

sb.append((char)i);}

loop=false;}

Thread.currentThread().sleep(50);}

//displaytheresponsetotheoutconsole

System.out.println(sb.toString());

socket.close();

注意要從web服務(wù)器得到正確的響應(yīng),你必須要發(fā)送用HTTP協(xié)議編譯了的HTTP請求。如果你看了上面的HTTP部分,你應(yīng)該能夠理解上面代碼中的HTTP請求。

編者注:這篇文章節(jié)選自budi自己出版的書<Tomcat內(nèi)幕>。你可以在他的網(wǎng)站得到更多的相關(guān)資料。

基于Java的Web服務(wù)器工作原理2

作者:fajaven譯發(fā)文時間:201*.09.1217:00:38

ServerSocket類

Socket類描述的是“客戶端”socket,當(dāng)你需要創(chuàng)建與遠(yuǎn)程服務(wù)程序連接時需要用到它。如果你想實現(xiàn)一個服務(wù)程序,如HTTP服務(wù)器或者FTP服務(wù)器,則需要另外不同的方法。這是因為你的服務(wù)器必須隨時服務(wù),它不知道什么時候會有一個客戶端程序需要連接它。

因為這個目的,你需要用到j(luò)ava.net.ServerSocket這個類,它是服務(wù)器端socket的一個實現(xiàn)。服務(wù)器端socket等待來自客戶端的連接請求。一旦它收到一個連接請求,它創(chuàng)建一個socket實例來與客戶端進(jìn)行通信。

要創(chuàng)建服務(wù)器端socket,需要用到ServerSocket類提供的四個構(gòu)建方法中的一個。你需要指定服務(wù)器端socket偵聽的IP地址與端口號。比較典型地,這個IP地址可以是127.0.0.1,意思是該服務(wù)器端socket偵聽的是本地機(jī)器。服務(wù)器端socket偵聽的IP地址指的是綁定地址。服務(wù)器端socket另一個重要的屬性是隊列長度,即它拒絕請求前所接受的最大請求排隊長度。

ServerSocket類的構(gòu)建方法之一如下:

publicServerSocket(intport,intbackLog,InetAddressbindingAddress);

對于這個構(gòu)建方法,綁定地址必須是java.net.InetAddress類的實例。創(chuàng)建一個InetAddress類的對象的簡單方法是呼叫其靜態(tài)方法getByName,傳遞一個包含主機(jī)名的字符串。

InetAddress.getByName("127.0.0.1");

以下行的代碼創(chuàng)建了一個服務(wù)器端socket,它偵聽本地機(jī)器的8080端口,限制隊列長度為1。

newServerSocket(8080,1,InetAddress.getByName("127.0.0.1"));

一旦有了一個ServerSocket實例,就可以通過呼叫其accept方法來讓它等待進(jìn)來的鏈接請求。這個方法只有當(dāng)接收到請求時才返回,它返回的是Socket類的實例。這個Socket對象就可以用來從客戶端應(yīng)用程序發(fā)送與接收字節(jié)流,正如上節(jié)據(jù)說的那樣。實際上,accept方法是本文例子中用到的唯一方法。

應(yīng)用實例

我們的web服務(wù)器程序是ex01.pyrmont包的一部分,它包含三個類:HttpServer;Request;Response。

整個程序的入口(靜態(tài)main方法)是HttpServer類。它創(chuàng)建一個HttpServer的實例,并呼叫其await方法。正如名字表達(dá)的,await在一個特定的端口等待HTTP請求,處理它們,并返回響應(yīng)給客戶端。它保持等待狀態(tài),直到收到停止命令。(用方法名await代替wait,是因為System中有一個重要的與線程相關(guān)的方法)

這個程序只從一個特定的目錄發(fā)送靜態(tài)資源,如HTML與圖像文件。它只支持沒有文件頭(如日期與cookie)的情況,F(xiàn)在我們將在如下的幾節(jié)中看一下這三個類。

HttpServer類

HttpServer實現(xiàn)了一個web服務(wù)器,它可以提供(serve)特定目錄及其子目錄下的靜態(tài)資源。這個特定的目錄由publicstaticfinalWEB_ROOT指定。

WEB_ROOT初始化如下:

publicstaticfinalStringWEB_ROOT=

System.getProperty("user.dir")+File.separator+"webroot";

代碼列表中包含了一具叫做webroot的目錄,里面有一些靜態(tài)的資源,你可以用來測試本應(yīng)用。你也可以看到一個servlet,在我的下一篇文章將會被用到:“Servlets容器是怎樣工作的”。

為了請求一個靜態(tài)的資源,在瀏覽器的地址欄輸入如是地址::port/staticResources

如果你從不同的機(jī)器上發(fā)送請求到運行本應(yīng)用的機(jī)器,則machinename是運行應(yīng)用機(jī)器的機(jī)器名或IP地址,port是8080,staticResources是被請求的文件名稱,它必須包含在WEB_ROOT目錄內(nèi)。

例如,如果你用同一臺電腦來測試這個應(yīng)用,你想要HttpServer發(fā)送index.html這個文件,用以下的地址::8080/index.html

要停止服務(wù),只需要從瀏覽器發(fā)送一個停止(shutdown)命令,即在瀏覽器的地址欄輸入host:port字段后,加上預(yù)先定義好的字符串。在我們的HttpServer類中,停止命令被定義為SHUTDOWN,一個staticfinal變量。

privatestaticfinalStringSHUTDOWN_COMMAND="/SHUTDOWN";

因此,要停止服務(wù),你可以這樣::8080/SHUTDOWN現(xiàn)在,讓我們看一下列表1.1中給出的await方法。代碼列表后面將對這段代碼做一些解釋。

Listing1.1.TheHttpServerclass"awaitmethod

publicvoidawait(){

ServerSocketserverSocket=null;

intport=8080;try{

serverSocket=newServerSocket(port,1,

InetAddress.getByName("127.0.0.1"));}

catch(IOExceptione){

e.printStackTrace();

System.exit(1);}

//Loopwaitingforarequest

while(!shutdown){

Socketsocket=null;

InputStreaminput=null;

OutputStreamoutput=null;try{

socket=serverSocket.accept();

input=socket.getInputStream();

output=socket.getOutputStream();

//createRequestobjectandparse

Requestrequest=newRequest(input);

request.parse();

//createResponseobject

Responseresponse=newResponse(output);

response.setRequest(request);

response.sendStaticResource();

//Closethesocket

socket.close();

//checkifthepreviousURIisashutdowncommand

shutdown=request.getUri().equals(SHUTDOWN_COMMAND);}

catch(Exceptione){

e.printStackTrace();

continue;}}}

await方法以創(chuàng)建一個ServerSocket實例開始,然后進(jìn)入一個while的循環(huán)。

serverSocket=newServerSocket(

port,1,InetAddress.getByName("127.0.0.1"));...

//Loopwaitingforarequest

while(!shutdown){...}

在while循環(huán)中的代碼,運行到ServerSocket的accept方法即停止。這個方法只有在8080端口接收到HTTP請求才返回:

socket=serverSocket.accept();

收到請求后,await方法從accept方法返回的Socket實例中等到j(luò)ava.io.InputStream與java.io.OutputStream:

input=socket.getInputStream();

output=socket.getOutputStream();

然后await方法創(chuàng)建一個Request對象,呼叫它的parse方法來解析這個原始的HTTP請求:

//createRequestobjectandparse

Requestrequest=newRequest(input);

request.parse();

下一步,await方法創(chuàng)建一個Response對象并把Request對象設(shè)置給它,呼叫它的sendStaticResource方法:

//createResponseobject

Responseresponse=newResponse(output);

response.setRequest(request);

response.sendStaticResource();

最后,await方法關(guān)閉Socket,呼叫Request的getUri方法來檢查HTTP請求的地址是否是一個停止命令。如果是,則shutdown變量被設(shè)置為true,程序退出while循環(huán):

//Closethesocket

socket.close();

//checkifthepreviousURIisashutdowncommand

shutdown=request.getUri().equals(SHUTDOWN_COMMAND);

基于Java的Web服務(wù)器工作原理3

作者:fajaven發(fā)文時間:201*.09.1217:11:54

Request類

Request類對應(yīng)HTTP請求。創(chuàng)建這個類的實例,并傳給它從Socket獲得的InputStream對象,從而捕獲與客戶端的通信。呼叫InputStream對象的read方法中的一個就可以得到HTTP請求的原始數(shù)據(jù)。

Request類有二個public方法parse與getUri。parse方法解析HTTP請求的原始數(shù)據(jù)。它做的事情不多--唯一它使之有效的信息是HTTP請求的URI,這個通過呼叫私有方法parseUri來獲得。parseUri方法把URI作為一個變量。調(diào)用getUri方法可以得到HTTP請求的URI。

要明白parse與parseUri的工作原理,你需要知道HTTP請求的結(jié)構(gòu),由RFC2616定義。

一個HTTP請求包括三個部分:Requestline;Headers;Messagebody。

現(xiàn)在,我們只需要關(guān)注HTTP請求的第一部分--請求行。請求行以方法記號開始,接著是請求的URI與協(xié)議版本,以回車換行符結(jié)束。請求行的元素之間以空格分開。例如,一個用GET方法的index.html文件的請求行如下:

GET/index.htmlHTTP/1.1

parse方法從socket的InputStream傳遞給Request對象中讀取字節(jié)流,把這個字節(jié)數(shù)組存在緩沖里。然后,它把buffer字節(jié)數(shù)組里的字節(jié)放入叫做request的StringBuffer對象中,再把StringBuffer替換成String傳遞給parseUri方法。

parse方法的代碼如列表1.2

Listing1.2.TheRequestclass"parsemethod

publicvoidparse(){

//ReadasetofcharactersfromthesocketStringBufferrequest=newStringBuffer(2048);inti;

byte[]buffer=newbyte[2048];try{

i=input.read(buffer);}

catch(IOExceptione){

e.printStackTrace();i=-1;}

for(intj=0;jrequest.append((char)buffer[j]);}

System.out.print(request.toString());

uri=parseUri(request.toString());}

parseUri方法查找請求行的第一個與第二個空格,從而從請求行獲得了URI。列表1.3展示了parseUri方法的代碼。

Listing1.3.TheRequestclass"parseUrimethod

privateStringparseUri(StringrequestString){intindex1,index2;

index1=requestString.indexOf("");

if(index1!=-1){

index2=requestString.indexOf("",index1+1);

if(index2>index1)

returnrequestString.substring(index1+1,index2);}

returnnull;}

Response類

Response類描述HTTP響應(yīng)。它的構(gòu)建方法接受OutputStream對象,如下:

publicResponse(OutputStreamoutput){

this.output=output;}

Response對象通過傳遞從socket獲得的OutputStream對象到HttpServer類的await方法而創(chuàng)建。

Response類有二個公共方法setRequest與setStaticResource。setRequest用來傳遞Request對象到Response對象。它比較簡單,代碼如列表1.4所示:

Listing1.4.TheResponseclass"setRequestmethod

publicvoidsetRequest(Requestrequest){

this.request=request;}

sendStaticResource方法用來發(fā)送靜態(tài)的資源,例如HTML文件。它的實現(xiàn)如列表1.5所示:

Listing1.5.TheResponseclass"sendStaticResourcemethod

publicvoidsendStaticResource()throwsIOException{

byte[]bytes=newbyte[BUFFER_SIZE];

FileInputStreamfis=null;try{

Filefile=newFile(HttpServer.WEB_ROOT,request.getUri());

if(file.exists()){

fis=newFileInputStream(file);

intch=fis.read(bytes,0,BUFFER_SIZE);

while(ch!=-1){

output.write(bytes,0,ch);

ch=fis.read(bytes,0,BUFFER_SIZE);}}else{

//filenotfound

StringerrorMessage="HTTP/1.1404FileNotFound\\r\\n"+

"Content-Type:text/html\\r\\n"+

"Content-Length:23\\r\\n"+"\\r\\n"+"

FileNotFound";

output.write(errorMessage.getBytes());}}

catch(Exceptione){

//thrownifcannotinstantiateaFileobject

System.out.println(e.toString());}finally{

if(fis!=null)

fis.close();}}

SendStaticResource方法非常簡單。它首先通過傳遞父與子目錄到File類的構(gòu)建方法從而實例化java.io.File類。

FilefilenewFile(HttpServer.WEB_ROOT,request.getUri());

然后檢查這個文件是否存在。如果存在,則sendStaticResource方法傳遞File對象創(chuàng)建java.io.FileInputStream對象。然后調(diào)用FileInputStream的read方法,并把字節(jié)數(shù)組寫到OutputStream對象output。就這樣,靜態(tài)資源的內(nèi)容作為原始數(shù)據(jù)被發(fā)送到瀏覽器。

if(file.exists()){

fis=newFileInputStream(file);

intch=fis.read(bytes,0,BUFFER_SIZE);

while(ch!=-1){

output.write(bytes,0,ch);

ch=fis.read(bytes,0,BUFFER_SIZE);}}

如果文件不存在,sendStaticResource發(fā)送一個錯誤信息到瀏覽器。

StringerrorMessage="HTTP/1.1404FileNotFound\\r\\n"+

"Content-Type:text/html\\r\\n"+

"Content-Length:23\\r\\n"+"\\r\\n"+"

FileNotFound";

output.write(errorMessage.getBytes());

編譯與運行應(yīng)用程序

要編輯與運行本文的應(yīng)用,首先你需要解壓源碼zip文件。直接解壓出來的目錄被稱為工作目錄,它有三個子目錄:src/,classes/,lib/。要編譯應(yīng)用,從工作目錄輸入如下命令:

javac-d.src/ex01/pyrmont/*.java

-d選項把結(jié)果寫到當(dāng)前目錄,而不是src/目錄。

要運行應(yīng)用,在當(dāng)前工作目錄輸入如下命令:

javaex01.pyrmont.HttpServer

測試這個應(yīng)用,打開你的瀏覽器,在地址欄輸入如下地址::8080/index.html

你將在你的瀏覽器看到index.html顯示出來,如圖1所示。

圖1:web服務(wù)器的輸出顯示

在控制臺,你看到如下的內(nèi)容:

GET/index.htmlHTTP/1.1

Accept:*/*Accept-Language:en-us

Accept-Encoding:gzip,deflate

User-Agent:Mozilla/4.0(compatible;MSIE4.01;Windows98)

Host:localhost:8080

Connection:Keep-Alive

GET/images/logo.gifHTTP/1.1

Accept:*/*

Referer::8080/index.html

Accept-Language:en-us

Accept-Encoding:gzip,deflate

User-Agent:Mozilla/4.0(compatible;MSIE4.01;Windows98)

Host:localhost:8080

Connection:Keep-Alive總結(jié)

在這篇文章中(分為三個部分),你看到了一個簡單的web服務(wù)器的工作原理。本文相關(guān)的應(yīng)用只包括了三個類,功能是不全面的。然而,它仍不失為一個好的學(xué)習(xí)工具。

(責(zé)任編輯:張明燕)

Java的網(wǎng)絡(luò)編程:用Java實現(xiàn)Web服務(wù)器

作者:谷和啟發(fā)文時間:201*.12.2615:35:57

HTTP協(xié)議

超文本傳輸協(xié)議(HTTP)是位于TCP/IP協(xié)議的應(yīng)用層,是最廣為人知的協(xié)議,也是互連網(wǎng)中最核心的協(xié)議之一,同樣,HTTP也是基于C/S或B/S模型實現(xiàn)的。事實上,我們使用的瀏覽器如Netscape或IE是實現(xiàn)HTTP協(xié)議中的客戶端,而一些常用的Web服務(wù)器軟件如Apache、IIS和iPlanetWebServer等是實現(xiàn)HTTP協(xié)議中的服務(wù)器端。Web頁由服務(wù)端資源定位,傳輸?shù)綖g覽器,經(jīng)過瀏覽器的解釋后,被客戶所看到。

Web的工作基于客戶機(jī)/服務(wù)器計算模型,由Web瀏覽器(客戶機(jī))和Web服務(wù)器(服務(wù)器)構(gòu)成,兩者之間采用超文本傳送協(xié)議(HTTP)進(jìn)行通信。HTTP協(xié)議是Web瀏覽器和Web服務(wù)器之間的應(yīng)用層協(xié)議,是通用的、無狀態(tài)的、面向?qū)ο蟮膮f(xié)議。

一個完整的HTTP協(xié)議會話過程包括四個步驟:

◆連接,Web瀏覽器與Web服務(wù)器建立連接,打開一個稱為Socket(套接字)的虛擬文件,此文件的建立標(biāo)志著連接建立成功;

◆請求,Web瀏覽器通過Socket向Web服務(wù)器提交請求。HTTP的請求一般是GET或POST命令(POST用于FORM參數(shù)的傳遞);

◆應(yīng)答,Web瀏覽器提交請求后,通過HTTP協(xié)議傳送給Web服務(wù)器。Web服務(wù)器接到后,進(jìn)行事務(wù)處理,處理結(jié)果又通過HTTP傳回給Web瀏覽器,從而在Web瀏覽器上顯示出所請求的頁面;

◆關(guān)閉連接,應(yīng)答結(jié)束后Web瀏覽器與Web服務(wù)器必須斷開,以保證其它Web瀏覽器能夠與Web服務(wù)器建立連接。

Java實現(xiàn)Web服務(wù)器功能的程序設(shè)計

編程思路

根據(jù)上述HTTP協(xié)議的會話過程,本實例中實現(xiàn)了GET請求的Web服務(wù)器程序的方法,方法如下:

通過創(chuàng)建ServerSocket類對象,偵聽用戶指定的端口(為8080),等待并接受客戶機(jī)請求到端口。創(chuàng)建與Socket相關(guān)聯(lián)的輸入流和輸出流,然后讀取客戶機(jī)的請求信息。若請求類型是GET,則從請求信息中獲取所訪問的HTML文件名;如果HTML文件存在,則打開HTML文件,把HTTP頭信息和HTML文件內(nèi)容通過Socket傳回給Web瀏覽器,然后關(guān)閉文件,否則發(fā)送錯誤信息給Web瀏覽器。最后關(guān)閉與相應(yīng)Web瀏覽器連接的Socket。

用Java編寫Web服務(wù)器httpServer.java文件的源代碼如下:

//httpServer.java

importjava.net.*;

importjava.io.*;

importjava.util.*;

importjava.lang.*;

publicclasshttpServer{

publicstaticvoidmain(Stringargs[]){intport;

ServerSocketserver_socket;

//讀取服務(wù)器端口號try{

port=Integer.parseInt(args[0]);}

catch(Exceptione){

port=8080;}try{

//監(jiān)聽服務(wù)器端口,等待連接請求

server_socket=newServerSocket(port);

System.out.println("httpServerrunningonport"+

server_socket.getLocalPort());

//顯示啟動信息

while(true){

Socketsocket=server_socket.accept();

System.out.println("Newconnectionaccepted"+

socket.getInetAddress()+

":"+socket.getPort());

//創(chuàng)建分線程try{

httpRequestHandlerrequest=

newhttpRequestHandler(socket);

Threadthread=newThread(request);

//啟動線程

thread.start();}

catch(Exceptione){

System.out.println;}}}

catch(IOExceptione){

System.out.println;}}}

classhttpRequestHandlerimplementsRunnable{

finalstaticStringCRLF="\\r\\n";

Socketsocket;

InputStreaminput;

OutputStreamoutput;

BufferedReaderbr;

//構(gòu)造方法

publichttpRequestHandler(Socketsocket)throwsException{

this.socket=socket;

this.input=socket.getInputStream();

this.output=socket.getOutputStream();

this.br=

newBufferedReader(newInputStreamReader(socket.getInputStream()));}

//實現(xiàn)Runnable接口的run()方法

publicvoidrun(){try{

processRequest();}

catch(Exceptione){

System.out.println;}}

privatevoidprocessRequest()throwsException{

while(true){

//讀取并顯示W(wǎng)eb瀏覽器提交的請求信息

StringheaderLine=br.readLine();

System.out.println("Theclientrequestis"+headerLine);

if(headerLine.equals(CRLF)||headerLine.equals(""))break;

StringTokenizers=newStringTokenizer(headerLine);

Stringtemp=s.nextToken();

if(temp.equals("GET")){

StringfileName=s.nextToken();

fileName="."+fileName;

//打開所請求的文件

FileInputStreamfis=null;

booleanfileExists=true;try{

fis=newFileInputStream(fileName);}

catch(FileNotFoundExceptione){

fileExists=false;}

//完成回應(yīng)消息

StringserverLine="Server:asimplejavahttpServer";

StringstatusLine=null;

StringcontentTypeLine=null;

StringentityBody=null;

StringcontentLengthLine="error";

if(fileExists){

statusLine="HTTP/1.0200OK"+CRLF;

contentTypeLine="Content-type:"+

contentType(fileName)+CRLF;

contentLengthLine="Content-Length:"

+(newInteger(fis.available())).toString()

+CRLF;}else{

statusLine="HTTP/1.0404NotFound"+CRLF;

contentTypeLine="text/html";

entityBody=""+""+

"404NotFound"+"

usage::port/"

+"fileName.html";}

//發(fā)送到服務(wù)器信息

output.write(statusLine.getBytes());

output.write(serverLine.getBytes());

output.write(contentTypeLine.getBytes());

output.write(contentLengthLine.getBytes());

output.write(CRLF.getBytes());//發(fā)送信息內(nèi)容

if(fileExists){

sendBytes(fis,output);

fis.close();}else{

output.write(entityBody.getBytes());}}}

//關(guān)閉套接字和流try{

output.close();

br.close();

socket.close();}

catch(Exceptione){}}

privatestaticvoidsendBytes(FileInputStreamfis,OutputStreamos)

throwsException{

//創(chuàng)建一個1Kbuffer

byte[]buffer=newbyte[1024];

intbytes=0;

//將文件輸出到套接字輸出流中

while((bytes=fis.read(buffer))!=-1){

os.write(buffer,0,bytes);}}

privatestaticStringcontentType(StringfileName){

if(fileName.endsWith(".htm")||fileName.endsWith(".html")){

return"text/html";}

return"fileName";}}

編程技巧說明◆主線程設(shè)計

主線程的設(shè)計就是在主線程httpServer類中實現(xiàn)了服務(wù)器端口的偵聽,服務(wù)器接受一個客戶端請求之后創(chuàng)建一個線程實例處理請求,代碼如下:

importjava.net.*;

importjava.io.*;

importjava.util.*;

importjava.lang.*;

publicclasshttpServer{

publicstaticvoidmain(Stringargs[]){port;

ServerSocketserver_socket;

//讀取服務(wù)器端口號try{

port=Integer.parseInt(args[0]);}

catch(Exceptione){

port=8080;}try{

//監(jiān)聽服務(wù)器端口,等待連接請求

server_socket=newServerSocket(port);

System.out.println("httpServerrunningonport"

+server_socket.getLocalPort());....................

◆連接處理分線程設(shè)計

在分線程httpRequestHandler類中實現(xiàn)了HTTP協(xié)議的處理,這個類實現(xiàn)了Runnable接口,代碼如下:

classhttpRequestHandlerimplementsRunnable{

finalstaticStringCRLF="\\r\\n";

Socketsocket;

InputStreaminput;

OutputStreamoutput;

BufferedReaderbr;

//構(gòu)造方法

publichttpRequestHandler(Socketsocket)throwsException{

this.socket=socket;

//得到輸入輸出流

this.input=socket.getInputStream();

this.output=socket.getOutputStream();

this.br=

newBufferedReader(newInputStreamReader(socket.getInputStream()));}

//實現(xiàn)Runnable接口的run()方法

publicvoidrun(){try{

processRequest();}

catch(Exceptione){

System.out.println;}}

◆構(gòu)建processRequest()方法來處理信息的接收和發(fā)送

作為實現(xiàn)Runnable接口的主要內(nèi)容,在run()方法中調(diào)用processRequest()方法來處理客戶請求內(nèi)容的接收和服務(wù)器返回信息的發(fā)送,代碼如下:

privatevoidprocessRequest()throwsException{

while(true){

//讀取并顯示W(wǎng)eb瀏覽器提交的請求信息

StringheaderLine=br.readLine();

System.out.println("Theclientrequestis"+headerLine);

if(headerLine.equals(CRLF)||headerLine.equals(""))break;

//根據(jù)請求字符串中的空格拆分客戶請求

StringTokenizers=newStringTokenizer(headerLine);

Stringtemp=s.nextToken();

if(temp.equals("GET")){

StringfileName=s.nextToken();

fileName="."+fileName;

.............

.............

在processRequest()方法中得到客戶端請求后,利用一個StringTokenizer類完成了字符串的拆分,這個類可以實現(xiàn)根據(jù)字符串中指定的分隔符(缺省為空格)將字符串拆分成為字串的功能。利用nextToken()方法依次得到這些字串;sendBytes()方法完成信息內(nèi)容的發(fā)送,contentType()方法用于判斷文件的類型。

顯示W(wǎng)eb頁面

顯示W(wǎng)eb頁面的index.html文件代碼如下:

*********歡迎你的到來!*********

這是一個用Java語言實現(xiàn)的Web服務(wù)器

--------------------------------------------------------------------------------

運行實例

為了測試上述程序的正確性,將編譯后的httpServer.class、httpRequestHandler.class和上面的index.html文件置于網(wǎng)絡(luò)的某臺主機(jī)的同一目錄中。

首先運行服務(wù)器程序javahttpServer8080,服務(wù)器程序運行后顯示端口信息“httpServerruningonport8080”,然后在瀏覽器的地址欄中輸入:8080/index.html,就可以正確顯示網(wǎng)頁,同時在顯示“httpServerruningonport8080”窗口中服務(wù)器會出現(xiàn)一些信息。

友情提示:本文中關(guān)于《JavaWeb工作原理》給出的范例僅供您參考拓展思維使用,JavaWeb工作原理:該篇文章建議您自主創(chuàng)作。

來源:網(wǎng)絡(luò)整理 免責(zé)聲明:本文僅限學(xué)習(xí)分享,如產(chǎn)生版權(quán)問題,請聯(lián)系我們及時刪除。


JavaWeb工作原理》由互聯(lián)網(wǎng)用戶整理提供,轉(zhuǎn)載分享請保留原作者信息,謝謝!
鏈接地址:http://www.hmlawpc.com/gongwen/585498.html
相關(guān)文章