Http客户端程序已集成在Java语言中,可以通过URLConnection类调用。遗憾的是,由于SUN没有公布Http客户程序的源码,它实现的细节仍是一个谜。本文根据HTTP协议规范,用Java.net.Socket类实现一个HTTP协议客户端程序。 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)。
3. HTTP客户端程序:
import java.net.*; import java.io.*; import java.util.Properties; import java.util.Enumeration; 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; try { if(client!=null) closeServer(); if(byteStream != null) { byteStream.close(); byteStream=null; } InetAddress address = InetAddress.getByName(host); client = new Socket(address,port==-1?80:port); sender = new BufferedOutputStream(client.getOutputStream()); receiver = new BufferedInputStream(client.getInputStream()); }catch(UnknownHostException u) { throw u; }catch(IOException i) { throw i; } } /* 关闭与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> responseCode = Integer.parseInt(message.substring(mark+1,mark+=4)); <br>responseMessage = message.substring(mark,message.length()).trim(); <br><br>// 应答状态码和处理请读者添加 <br>switch(responseCode) { <br>case 400: <br>throw new IOException("错误请求"); <br>case 404: <br>throw new FileNotFoundException( getURLFormat(target) ); <br>case 503: <br>throw new IOException("服务器不可用" ); <br>} <br>if(word==-1) throw new ProtocolException("信息接收异常终止"); <br>int symbol=-1; <br>count=0; <br>// 解析元信息 <br>while( word!='/r' && word!='/n' && word>-1) { <br>if(word=='/t') word=32; <br>if(count==data.length) data = addCapacity(data); <br>data[count++] = (byte)word; <br>parseLine: { <br>while( (symbol=receiver.read()) >-1 ) { <br>switch(symbol) { <br>case '/t': <br>symbol=32; break; <br>case '/r': <br>case '/n': <br>word = receiver.read(); <br>if( symbol=='/r' && word=='/n') { <br>word=receiver.read(); <br>if(word=='/r') word=receiver.read(); <br>} <br>if( word=='/r' || word=='/n' || word>32) break parseLine; <br>symbol=32; break; <br>} <br>if(count==data.length) data = addCapacity(data); <br>data[count++] = (byte)symbol; <br>} <br>word=-1; <br>} <br>message = new String(data,0,count); <br>mark = message.indexOf(':'); <br>String key = null; <br>if(mark>0) key = message.substring(0,mark); <br>mark++; <br>while( mark<message.length mark message.charat> String value = message.substring(mark,message.length() ); <br>header.put(key,value); <br>count=0; <br>} <br>// 获得正文数据 <br>while( (word=receiver.read())!=-1) { <br>if(count == data.length) data = addCapacity(data); <br>data[count++] = (byte)word; <br>} <br>if(count>0) byteStream = new ByteArrayInputStream(data,0,count); <br>data=null; <br>closeServer(); <br>} <br>public String getResponseMessage() { <br>return responseMessage; <br>} <br>public int getResponseCode() { <br>return responseCode; <br>} <br>public String getServerVersion() { <br>return serverVersion; <br>} <br>public InputStream getInputStream() { <br>return byteStream; <br>} <br>public synchronized String getHeaderKey(int i) { <br>if(i>=header.size()) return null; <br>Enumeration enum = header.propertyNames(); <br>String key = null; <br>for(int j=0; j<=i; j++) <br>key = (String)enum.nextElement(); <br>return key; <br>} <br>public synchronized String getHeaderValue(int i) { <br>if(i>=header.size()) return null; <br>return header.getProperty(getHeaderKey(i)); <br>} <br>public synchronized String getHeaderValue(String key) { <br>return header.getProperty(key); <br>} <br>protected String getBaseHeads() { <br>String inf = "User-Agent: myselfHttp/1.0/r/n"+ <br>"Accept: www/source; text/html; image/gif; */*/r/n"; <br>return inf; <br>} <br>private byte[] addCapacity(byte rece[]){ <br>byte temp[] = new byte[rece.length+1024]; <br>System.arraycopy(rece,0,temp,0,rece.length); <br>return temp; <br>} <br>} <br></message.length></message.length> |
相关推荐
用socket 类实现http 客户端应用 Http客户端程序已集成在Java语言中,可以通过URLConnection类调用。遗憾的 是,由于SUN没有公布Http客户程序的源码,它实现的细节仍是一个谜。本文根据HTTP 协议规范,用Java.net....
用Socket套接字实现DAYTIME协议的服务器和客户端程序.doc
Http客户端程序已集成在Java语言中,可以通过URLConnection类调用。遗憾的是,由于SUN没有公布Http客户程序的源码,它实现的细节仍是一个谜...本文根据HTTP协议规范,用Java.net.Socket类实现一个HTTP协议客户端程序。
矿用产量数据采集应用实例,使用HPsocket框架结构,通过UDP协议接收从服务器上传的数据并进行解码
通过使用Socket类,开发人员可以创建客户端和服务器端应用程序,并实现它们之间的双向通信。 使用Socket类进行网络编程的基本模式是: 创建一个Socket对象,并指定要使用的协议类型(如TCP或UDP)。 使用Socket...
根据自定义的协议规范,使用 Socket 编程接口编写基本的网络应用软件。 掌握 C 语言形式的 Socket 编程接口用法,能够正确发送和接收网络数据包。 开发一个客户端,实现人机交互界面和与服务器的通信。 开发一个...
本实验通过一个简单的网络应用来实现客户端与服务器端的通信.客户端给服务器端发送一行字符串,服务器接收字符串,同时可以发回复给客户端,在客户端显示出来。 在Socket通信中,发送方和接收方的两个Socket之间必须...
实现测试代码:使用Socket编程接口,编写客户端和服务器端的测试代码。这些代码应包含测试用例中涉及的所有操作。 运行测试:启动客户端和服务器端程序,进行实际测试。可以通过手动或自动化方式发送测试数据,并...
参考该文章的代码http://www.cnblogs.com/yuqiao/p/5786427.html.socket是在应用层和传输层之间的一个抽象层,它把TCP/IP层复杂的操作抽象为几个简单的接口供应用层调用已实现进程在网络中通信。 socket起源于UNIX...
应用的协议头 发送应用程序消息 应用程序消息类 消息入口点 服务端主程序 服务端消息类 客户端主程序 客户端消息类 消息类的包装 运行客户端 / 服务器应用程序 故障排查 ping netstat windows Wireshark 引用 Python...
实验的目的: 1) 掌握Java Socket编程应用; 2) 阅读HTTP协议相关规范; 3) 基于Java Socket构建简单的HTTP的客户端和服务器; 4) 构建支持并发的HTTP服务器。
根据自定义的协议规范,使用 Socket 编程接口编写基本的网络应用软件。 掌握 C 语言形式的 Socket 编程接口用法,能够正确发送和接收网络数据包 开发一个客户端,实现人机交互界面和与服务器的通信 开发一个服务端,...
文件时简单的UDP服务器程序和UDP客户端程序的例子,这两个程序都是Win32应用程序,在本地机器上的8888端口进行通信
一个socket的网络应用实例,能实现两个客户端之间的通讯!适合于嵌入式的应用
如果你熟悉 Winsock API函数,那么用Socket类编写网络程序会非常容易,当然,如果你不曾接触过,也不会太困难,跟随下面的解说,你会发觉使用 Socket类开发windows 网络应用程序原来有规可寻,它们在大多数情况下...
通过ReciveFrom()方法接收指定主机发送的消息(需提供主机IP地址及端口)下面用代码实现简单的服务器---客户端通信服务器端:客户端: 执行过程:(1)启动服务器(2)启动客户端,并向服务器发送数据分
wampy.js, 简单 WAMP ( web socket应用程序消息传递协议) Javascript实现 wampy.js简单 WAMP ( web socket应用程序消息传递协议) Javascript客户端实现 目录描述用法示例安装工具更新版本API插件构造函数选项...
另一种是针对client/server(客户端/服务器)模式的应用以及实现某些特殊的协议的应用,它的通讯过程是基于TCP/IP协议中传输层接口socket实现的。本文想简单的介绍一下Socket编程的Java实现方法。
VS2010 同步TCP应用编程---“吃棋子”游戏 服务器+客户端 ,学习服务器客户端进阶例程 利用TCP协议和同步套接字以及多线程编写 在监听线程中,服务器一旦接受一个连接,就创建一个与该玩家对应的线程,用于接收该...