| 巍 的个人资料fantasyswall照片日志列表 | 帮助 |
|
5月30日 HtmlBrowserpackage com.test.hello; import java.io.IOException; public class HtmlBrowser extends JFrame { public HtmlBrowser() { // 定义构造方法 private void jbInit() throws Exception { // 界面初始化 void textFieldURL_actionPerformed(ActionEvent e) { // 输入地址后响应回车 void jEditorPane_hyperlinkUpdate(HyperlinkEvent e) { // 响应页面打开超链接消息 protected void processWindowEvent(WindowEvent e) { // 处理窗体事件 public static void main(String[] args) { // Main函数 5月23日 java client for httppackage com.test.hello;
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> */ 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);
}
public static void main(String[] args){
Http http = new Http("http://localhost:8080/test/index.html"); System.out.println(http.getResponseMessage()); } /* 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;
}
}
|
|
|