分享
 
 
 

Http.java

王朝java/jsp·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

import java.net.*;

import java.io.*;

import java.util.Properties;

import java.util.Enumeration;

/**

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

**空行**

......

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

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

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

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

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

*/

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

??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' || word32) 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(mark0) key = message.substring(0,mark);

???mark++;

???while( 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(count0) 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 enum = header.propertyNames();

??String key = null;

??for(int j=0; j

???key = (String)enum.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: IcewolfHttp/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;

?}

?public static void main(String [] args){

??/*

??Http p = new Http();

??p.GET("http://www.google.com/search?q=java+book");

??byte[] bGets = new byte[1024];

??ByteArrayInputStream pS = (ByteArrayInputStream)p.getInputStream();

??int offset = 0;

??while((offset=pS.read(bGets,0,1024)) != -1){

???System.out.print(new String(bGets,0,offset));

??}

??*/

??String target = "http://www.google.com/search?q=java+book";

??try {

???URL url = new URL(target);

???HttpURLConnection pconn = (HttpURLConnection)url.openConnection();

???//HttpURLConnection.setFollowRedirects(true);

???//pconn.setInstanceFollowRedirects(true);

???pconn.addRequestProperty("User-Agent","IcewolfHttp/1.0");

???pconn.addRequestProperty("Accept","www/source; text/html; image/gif; */*");

???

???pconn.connect();

???System.out.println("Connect status:"+pconn.getResponseCode());

???//if(HttpURLConnection.HTTP_ACCEPTED == pconn.getResponseCode())

???//InputStream in = url.openConnection();

???InputStream in = pconn.getInputStream();

???System.out.println("Get status:"+pconn.getResponseCode());

???BufferedInputStream buff = new BufferedInputStream(in);

???Reader r = new InputStreamReader(buff);

???int c = 0;

???while ( (c = r.read()) != -1){

????System.out.print((char)c);

???}

???buff.close();

???in.close();

???pconn.disconnect();

??} catch (MalformedURLException mfe) {

???System.err.println(target + " is not a parsable URL");

??} catch (IOException ioe) {

???System.err.println(ioe);

??}

?}

}

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有