您的位置:首页 > 理论基础 > 计算机网络

Http协议客户端的JAVA简单实现

2011-02-04 21:44 746 查看
import java.net.*;

import java.io.*;

import java.util.Properties;

import java.util.Enumeration;

/**

Http客户端程序已集成在Java语言中,可以通过URLConnection类调用。遗憾的

是,由于SUN没有公布Http客户程序的源码,它实现的细节仍是一个谜。本文根据HTTP

协议规范,用Java.net.Socket类实现一个HTTP协议客户端程序.

<pre>

1.Socket类:

了解TCP/IP协议集通信的读者知道,协议间的通信是通过Socket完成的。在

Java.net包中,Socket类就是对Socket的具体实现。它通过连接到主机后,返回一个

I/O流,实现协议间的信息交换。

2 . HTTP协议

HTTP协议同其它TCP/IP协议集中的协议一样,是遵循客户/服务器模型工作的。客

户端发往服务端的信息格式如下:

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

请求方法 URL HTTP协议的版本号

提交的元信息

**空行**

实体

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

请求方法是对这次连接工作的说明,目前HTTP协议已经发展到1.1版,它包括GET、

HEAD、POST、DELETE、OPTIONS、TRACE、PUT七种。元信息是关于当前请求的信息。通

过分析元信息,可以检查实体数据是否完整,接收过程是否出错,类型是否匹配等。元

信息的引入使HTTP协议通信更加稳妥可靠。实体是请求的具体内容。

将上述报文发往Web服务器,如果成功,应答格式如下:

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

HTTP协议的版本号 应答状态码 应答状态码说明

接收的元信息

**空行**

实体

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

以上报文发向客户端,并且接收成功,彼此间关闭连接,完成一次握手。

下面用最常用的GET方法,来说明具体的报文应用

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

GET http://www.youhost.com HTTP/1.0

accept: www/source; text/html; image/gif; image/jpeg; */*

User_Agent: myAgent

**空行**

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

这个报文是向www.youhost.com主机请求一个缺省HTML文档。客户端HTTP协议版本

号是1.0版,元信息包括可接收的文件格式,用户代理,每一段之间用回车换行符分

隔,最后以一个空行结束。发向服务器后,如果执行过程正常,服务器返回以下代码:

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

HTTP/1.1 200 OK

Date: Tue, 14 Sep 1999 02:19:57 GMT

Server: Apache/1.2.6

Connection: close

Content-Type: text/html

**空行**

<html><head>...</head><body>...</body></html>

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

HTTP/1.1表示这个HTTP服务器是1.1版,200是服务器对客户请求的应答状态码,OK

是对应答状态码的解释,之后是这个文档的元信息和文档正文。(相关应答状态码和元

信息的解释请参阅Inetrnet标准草案:RFC2616)。

注: 程序中只实现GET、HEAD、POST三种方法。其他几种因不常使用,暂且忽略。

</pre>

*/

public class Http{

protected Socket client;

protected BufferedOutputStream sender;

protected BufferedInputStream receiver;

protected ByteArrayInputStream byteStream;

protected URL target;

private int responseCode = -1;

private String responseMessage = "";

private String serverVersion = "";

private Properties header = new Properties();

public Http(){}

public Http(String url){

GET(url);

}

/* GET方法根据URL,会请求文件、数据库查询结果、程序运行结果等多种内容 */

public void GET(String url){

try{

checkHTTP(url);

openServer(target.getHost(),target.getPort());

String cmd = "GET " + getURLFormat(target) + " HTTP/1.0/r/n" +

getBaseHeads() + "/r/n";

sendMessage(cmd);

receiveMessage();

}

catch(ProtocolException p){

p.printStackTrace();

return;

}

catch(UnknownHostException e){

e.printStackTrace();

return;

}

catch(IOException i){

i.printStackTrace();

return;

}

}

/*

* HEAD方法只请求URL的元信息,不包括URL本身。若怀疑本机和服务器上的

* 文件相同,用这个方法检查最快捷有效。

*/

public void HEAD(String url){

try{

checkHTTP(url);

openServer(target.getHost(),target.getPort());

String cmd = "HEAD " + getURLFormat(target) + " HTTP/1.0/r/n" +

getBaseHeads() + "/r/n";

sendMessage(cmd);

receiveMessage();

}

catch(ProtocolException p){

p.printStackTrace();

return;

}

catch(UnknownHostException e){

e.printStackTrace();

return;

}

catch(IOException i){

i.printStackTrace();

return;

}

}

/*

* POST方法是向服务器传送数据,以便服务器做出相应的处理。例如网页上常用的

* 提交表格。

*/

public void POST(String url,String content){

try{

checkHTTP(url);

openServer(target.getHost(),target.getPort());

String cmd = "POST " + getURLFormat(target) + " HTTP/1.0/r/n" +

getBaseHeads();

cmd += "Content-type: application/x-www-form-urlencoded/r/n";

cmd += "Content-length: " + content.length() + "/r/n/r/n";

cmd += content + "/r/n";

sendMessage(cmd);

receiveMessage();

}

catch(ProtocolException p){

p.printStackTrace();

return ;

}

catch(UnknownHostException e){

e.printStackTrace();

return ;

}

catch(IOException i){

i.printStackTrace();

return ;

}

}

protected void checkHTTP(String url) throws ProtocolException{

try{

URL target = new URL(url);

if(target == null ||

!target.getProtocol().toUpperCase().equals("HTTP")){

throw new ProtocolException("这不是HTTP协议");

}

this.target = target;

}

catch(MalformedURLException m){

throw new ProtocolException("协议格式错误");

}

}

/*

* 与Web服务器连接。若找不到Web服务器,InetAddress会引发UnknownHostException

* 异常。若Socket连接失败,会引发IOException异常。

*/

protected void openServer(String host,int port) throws UnknownHostException,

IOException{

header.clear();

responseMessage = "";

responseCode = -1;

if(client != null){

closeServer();

}

if(byteStream != null){

byteStream.close();

byteStream = null;

}

InetAddress address = InetAddress.getByName(host);

client = new Socket(address,port == -1 ? 80 : port);

client.setSoTimeout(5000);

sender = new BufferedOutputStream(client.getOutputStream());

receiver = new BufferedInputStream(client.getInputStream());

}

/* 关闭与Web服务器的连接 */

protected void closeServer() throws IOException{

if(client == null){

return;

}

try{

client.close();

sender.close();

receiver.close();

}

catch(IOException i){

throw i;

}

client = null;

sender = null;

receiver = null;

}

protected String getURLFormat(URL target){

String spec = "http://" + target.getHost();

if(target.getPort() != -1){

spec += ":" + target.getPort();

}

return spec += target.getFile();

}

/* 向Web服务器传送数据 */

protected void sendMessage(String data) throws IOException{

sender.write(data.getBytes(),0,data.length());

sender.flush();

}

/* 接收来自Web服务器的数据 */

protected void receiveMessage() throws IOException{

byte data[] = new byte[1024];

int count = 0;

int word = -1;

// 解析第一行

while( (word = receiver.read()) != -1){

if(word == '/r' || word == '/n'){

word = receiver.read();

if(word == '/n') {

word = receiver.read();

}

break;

}

if(count == data.length) {

data = addCapacity(data);

}

data[count++] = (byte) word;

}

String message = new String(data,0,count);

int mark = message.indexOf(32);

serverVersion= message.substring(0,mark);

while(mark < message.length() && message.charAt(mark + 1) == 32) {

mark++;

}

responseCode = Integer.parseInt(message.substring(mark + 1,mark += 4));

responseMessage = message.substring(mark,message.length()).trim();

// 应答状态码和处理请读者添加

switch(responseCode){

case 400:

throw new IOException("错误请求");

case 404:

throw new FileNotFoundException(getURLFormat(target));

case 503:

throw new IOException("服务器不可用");

}

if(word == -1){

throw new ProtocolException("信息接收异常终止");

}

int symbol = -1;

count = 0;

// 解析元信息

while(word != '/r' && word != '/n' && word > -1){

if(word == '/t') {

word = 32;

}

if(count == data.length) {

data = addCapacity(data);

}

data[count++] = (byte) word;

parseLine:{

while( (symbol = receiver.read()) > -1){

switch(symbol){

case '/t':

symbol = 32;

break;

case '/r':

case '/n':

word = receiver.read();

if(symbol == '/r' && word == '/n'){

word = receiver.read();

if(word == '/r') {

word = receiver.read();

}

}

if(word == '/r' || word == '/n' || word > 32){

break parseLine;

}

symbol = 32;

break;

}

if(count == data.length) {

data = addCapacity(data);

}

data[count++] = (byte) symbol;

}

word = -1;

}

message = new String(data,0,count);

mark = message.indexOf(':');

String key = null;

if(mark > 0) {

key = message.substring(0,mark);

}

mark++;

while(mark < message.length() && message.charAt(mark) <= 32) {

mark++;

}

String value = message.substring(mark,message.length());

header.put(key,value);

count = 0;

}

// 获得正文数据

while( (word = receiver.read()) != -1){

if(count == data.length) {

data = addCapacity(data);

}

data[count++] = (byte) word;

}

if(count > 0) {

byteStream = new ByteArrayInputStream(data,0,count);

}

data = null;

closeServer();

}

public String getResponseMessage(){

return responseMessage;

}

public int getResponseCode(){

return responseCode;

}

public String getServerVersion(){

return serverVersion;

}

public InputStream getInputStream(){

return byteStream;

}

public synchronized String getHeaderKey(int i){

if(i >= header.size()){

return null;

}

Enumeration enumss = header.propertyNames();

String key = null;

for(int j = 0; j <= i; j++){

key = (String) enumss.nextElement();

}

return key;

}

public synchronized String getHeaderValue(int i){

if(i >= header.size()){

return null;

}

return header.getProperty(getHeaderKey(i));

}

public synchronized String getHeaderValue(String key){

return header.getProperty(key);

}

protected String getBaseHeads(){

String inf = "User-Agent: ZealHttp/1.0/r/nAccept: www/source; text/html; image/gif; */*/r/n";

return inf;

}

private byte[] addCapacity(byte rece[]){

byte temp[] = new byte[rece.length + 1024];

System.arraycopy(rece,0,temp,0,rece.length);

return temp;

}

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: