[來(lái) 源:?電腦報(bào) ]??????????[發(fā)表日期: 2005-6-21]
一、HTTP協(xié)議的作用原理
HTTP協(xié)議的作用原理包括四個(gè)步驟:
??????? 1.連接:Web瀏覽器與Web服務(wù)器建立連接。2.請(qǐng)求:Web瀏覽器通過(guò)socket向Web服務(wù)器提交請(qǐng)求。3.應(yīng)答:Web瀏覽器提交請(qǐng)求后,通過(guò)HTTP傳送給Web服務(wù)器。Web服務(wù)器接到請(qǐng)求后,進(jìn)行事務(wù)處理,處理結(jié)果又通過(guò)HTTP傳回給Web瀏覽器,從而在Web瀏覽器上顯示出所請(qǐng)求的頁(yè)面。4.關(guān)系連接:當(dāng)應(yīng)答結(jié)束后,Web瀏覽器與Web服務(wù)器必須斷開,以保證其它Web瀏覽器能夠與Web服務(wù)器建立連接。
二、用Java實(shí)現(xiàn)Web服務(wù)器的程序設(shè)計(jì)
根據(jù)上述HTTP協(xié)議的作用原理,實(shí)現(xiàn)GET請(qǐng)求的Web服務(wù)器程序的方法如下:
??????? 1.創(chuàng)建ServerSocket類對(duì)象,監(jiān)聽端口8080。這是為了區(qū)別于HTTP的標(biāo)準(zhǔn)TCP/IP端口80而取的;2.等待、接受客戶機(jī)連接到端口8080,得到與客戶機(jī)連接的socket;3.創(chuàng)建與socket關(guān)聯(lián)的輸入流instream和輸入出流outstream;
式為:GET路徑/文件名HTTP/1.0;4.從與socket關(guān)聯(lián)的輸入流instream中讀取一行客戶機(jī)提交的請(qǐng)求信息,請(qǐng)求信息的格式為:GET路徑/文件名HTTP/1.0;5.從請(qǐng)求信息中獲取請(qǐng)求類型。如果請(qǐng)求類型是GET,則從請(qǐng)求信息中獲取所訪問(wèn)的HTML文件名。沒(méi)有HTML文件名時(shí),則以index.htm1作為文件名;6.如果HTML文件存在,則打開HTML文件,把HTTP頭信息和HTML文件內(nèi)容通過(guò)socket傳回給Web服務(wù)器,然后關(guān)閉文件,否則發(fā)送錯(cuò)誤信息給Web瀏覽器;7.關(guān)閉與相應(yīng)Web瀏覽器連接的socket字。
下面的程序是根據(jù)上述方法編寫的,可實(shí)現(xiàn)多線程的Web服務(wù)器,以保證多個(gè)客戶機(jī)能同時(shí)與該Web服務(wù)器連接。
//WebServer.java用Java編寫Web服務(wù)器
import java.io.*;
import java.net.*;
import java.util.Date;
public class WebServer{
public static void main(String args[])
{
int i=1,PORT=8080;
ServerSocket server=null;
Socketclient=null;
try{
server=new ServerSocket(PORT);
System.out.println
("Web Server is listening on port"
+server.getLocalPort());
for(;;){
client=server.accept();
//接受客戶機(jī)的連接請(qǐng)求
new Connection Thread(client,i).start();
i++;
}
}catch(Exception e){System.out.println(e);}
}
}
/*Connnection Thread類完成
與一個(gè)Web瀏覽器的通信*/
class Connection Thread extends Thread{
Socket client;//連接Web瀏覽器的socket字
int counter;//計(jì)數(shù)器
public Connection Thread(Socketcl,int c){
client=cl;
counter=c;
}
public void run()//線程體
{
try{
String deskIP=client.getInetAddress().toString();
//客戶機(jī)IP地址
int destport=client.getPort();
//客戶機(jī)端口號(hào)
System.out.println
("Connecction"+counter+":
connected to "+destIP+"on port
"+destport+".");
PrintStream outstream=new printStream
(client.getOoutputStream());
DataInputStreaminstream+new DataInputStream
(client.getInputStream());
String inline=instream.readLine();
//讀取Web瀏覽器提交的請(qǐng)求信息
System.out.println("Received:"+inline);
if(getrequest(inline)){//如果是GET請(qǐng)求
String filename=getfilename(inline);
File file=new File (filename);
if(file.exists()){
//若文件存在,則將文件送給Web瀏覽器
System.out.println(filename+"requested.");
outstream.println("HTTP/1.0200OK");
outstream.println("MIME_version:1.0");
outstream.println("Content_Type:text/htm1");
int len=(int)file.length();
outstream.println("Content_Length:"+len);
outstream.println("");
sendfile(outstream,file);//發(fā)送文件
outstream.flush();
}else{//文件不存在時(shí)
String notfound="htmlheadtitle
Not Found/title/head
bodyhlError404-File notfound
/hl/body/html";
outstream.println("HTTP /1.0 404 no found");
outstream.println("Content_Type:text /html");
outstream.println
("Content_Length:" +notfound.length() +2);
outstream.println("");
outstream.println(notfound);
outstream.flush();
}
}
long m1=1;
while(m10)
{
if(s.substring(0,3).equalsIgnoreCase
("GET"))return true;
}
return false;
}
/*獲取要訪問(wèn)的文件名*/
String getfilename(String s){
String f=s.substring(s.indexOf()+1);
f=f.substring(0,f.indexOf());
try{
if(f.charAt(0)==/)
f=f.substring(1);
}catch(String IndexOutOfBoundsException e){
System.out.println("Exception:"+e);
}
if(f.equals(""))f="index.html";
return f;
}
/*把指定文件發(fā)送給Web瀏覽器*/
void sendfile(PrintStream outs,File file){
try{
DataInputStreamin=new DataInputStream
(new FileInputStream(file));
int len=(int)file.length();
byte buf[]=new byte[len];
in.readFully(buf);
outs.write(buf,0,len);
outs.flush();
in.close();
}catch(Exception e){
System.out.println("Error retrieving file.");
System.exit(1);
}
}
}
程序中的Connection Thread線程子類用來(lái)分析一個(gè)Web瀏覽器提交的請(qǐng)求,并將應(yīng)答信息傳回給Web瀏覽器。其中,getrequest()方法用來(lái)檢測(cè)客戶的請(qǐng)求是否為"GET";getfilename(s)方法是從客戶請(qǐng)求信息s中獲取要訪問(wèn)的HTML文件名;sendfile()方法把指定文件內(nèi)容通過(guò)socket傳回給Web瀏覽器。
HTTP協(xié)議的作用原理包括四個(gè)步驟:
??????? 1.連接:Web瀏覽器與Web服務(wù)器建立連接。2.請(qǐng)求:Web瀏覽器通過(guò)socket向Web服務(wù)器提交請(qǐng)求。3.應(yīng)答:Web瀏覽器提交請(qǐng)求后,通過(guò)HTTP傳送給Web服務(wù)器。Web服務(wù)器接到請(qǐng)求后,進(jìn)行事務(wù)處理,處理結(jié)果又通過(guò)HTTP傳回給Web瀏覽器,從而在Web瀏覽器上顯示出所請(qǐng)求的頁(yè)面。4.關(guān)系連接:當(dāng)應(yīng)答結(jié)束后,Web瀏覽器與Web服務(wù)器必須斷開,以保證其它Web瀏覽器能夠與Web服務(wù)器建立連接。
二、用Java實(shí)現(xiàn)Web服務(wù)器的程序設(shè)計(jì)
根據(jù)上述HTTP協(xié)議的作用原理,實(shí)現(xiàn)GET請(qǐng)求的Web服務(wù)器程序的方法如下:
??????? 1.創(chuàng)建ServerSocket類對(duì)象,監(jiān)聽端口8080。這是為了區(qū)別于HTTP的標(biāo)準(zhǔn)TCP/IP端口80而取的;2.等待、接受客戶機(jī)連接到端口8080,得到與客戶機(jī)連接的socket;3.創(chuàng)建與socket關(guān)聯(lián)的輸入流instream和輸入出流outstream;
式為:GET路徑/文件名HTTP/1.0;4.從與socket關(guān)聯(lián)的輸入流instream中讀取一行客戶機(jī)提交的請(qǐng)求信息,請(qǐng)求信息的格式為:GET路徑/文件名HTTP/1.0;5.從請(qǐng)求信息中獲取請(qǐng)求類型。如果請(qǐng)求類型是GET,則從請(qǐng)求信息中獲取所訪問(wèn)的HTML文件名。沒(méi)有HTML文件名時(shí),則以index.htm1作為文件名;6.如果HTML文件存在,則打開HTML文件,把HTTP頭信息和HTML文件內(nèi)容通過(guò)socket傳回給Web服務(wù)器,然后關(guān)閉文件,否則發(fā)送錯(cuò)誤信息給Web瀏覽器;7.關(guān)閉與相應(yīng)Web瀏覽器連接的socket字。
下面的程序是根據(jù)上述方法編寫的,可實(shí)現(xiàn)多線程的Web服務(wù)器,以保證多個(gè)客戶機(jī)能同時(shí)與該Web服務(wù)器連接。
//WebServer.java用Java編寫Web服務(wù)器
import java.io.*;
import java.net.*;
import java.util.Date;
public class WebServer{
public static void main(String args[])
{
int i=1,PORT=8080;
ServerSocket server=null;
Socketclient=null;
try{
server=new ServerSocket(PORT);
System.out.println
("Web Server is listening on port"
+server.getLocalPort());
for(;;){
client=server.accept();
//接受客戶機(jī)的連接請(qǐng)求
new Connection Thread(client,i).start();
i++;
}
}catch(Exception e){System.out.println(e);}
}
}
/*Connnection Thread類完成
與一個(gè)Web瀏覽器的通信*/
class Connection Thread extends Thread{
Socket client;//連接Web瀏覽器的socket字
int counter;//計(jì)數(shù)器
public Connection Thread(Socketcl,int c){
client=cl;
counter=c;
}
public void run()//線程體
{
try{
String deskIP=client.getInetAddress().toString();
//客戶機(jī)IP地址
int destport=client.getPort();
//客戶機(jī)端口號(hào)
System.out.println
("Connecction"+counter+":
connected to "+destIP+"on port
"+destport+".");
PrintStream outstream=new printStream
(client.getOoutputStream());
DataInputStreaminstream+new DataInputStream
(client.getInputStream());
String inline=instream.readLine();
//讀取Web瀏覽器提交的請(qǐng)求信息
System.out.println("Received:"+inline);
if(getrequest(inline)){//如果是GET請(qǐng)求
String filename=getfilename(inline);
File file=new File (filename);
if(file.exists()){
//若文件存在,則將文件送給Web瀏覽器
System.out.println(filename+"requested.");
outstream.println("HTTP/1.0200OK");
outstream.println("MIME_version:1.0");
outstream.println("Content_Type:text/htm1");
int len=(int)file.length();
outstream.println("Content_Length:"+len);
outstream.println("");
sendfile(outstream,file);//發(fā)送文件
outstream.flush();
}else{//文件不存在時(shí)
String notfound="htmlheadtitle
Not Found/title/head
bodyhlError404-File notfound
/hl/body/html";
outstream.println("HTTP /1.0 404 no found");
outstream.println("Content_Type:text /html");
outstream.println
("Content_Length:" +notfound.length() +2);
outstream.println("");
outstream.println(notfound);
outstream.flush();
}
}
long m1=1;
while(m10)
{
if(s.substring(0,3).equalsIgnoreCase
("GET"))return true;
}
return false;
}
/*獲取要訪問(wèn)的文件名*/
String getfilename(String s){
String f=s.substring(s.indexOf()+1);
f=f.substring(0,f.indexOf());
try{
if(f.charAt(0)==/)
f=f.substring(1);
}catch(String IndexOutOfBoundsException e){
System.out.println("Exception:"+e);
}
if(f.equals(""))f="index.html";
return f;
}
/*把指定文件發(fā)送給Web瀏覽器*/
void sendfile(PrintStream outs,File file){
try{
DataInputStreamin=new DataInputStream
(new FileInputStream(file));
int len=(int)file.length();
byte buf[]=new byte[len];
in.readFully(buf);
outs.write(buf,0,len);
outs.flush();
in.close();
}catch(Exception e){
System.out.println("Error retrieving file.");
System.exit(1);
}
}
}
程序中的Connection Thread線程子類用來(lái)分析一個(gè)Web瀏覽器提交的請(qǐng)求,并將應(yīng)答信息傳回給Web瀏覽器。其中,getrequest()方法用來(lái)檢測(cè)客戶的請(qǐng)求是否為"GET";getfilename(s)方法是從客戶請(qǐng)求信息s中獲取要訪問(wèn)的HTML文件名;sendfile()方法把指定文件內(nèi)容通過(guò)socket傳回給Web瀏覽器。