`
school104
  • 浏览: 73304 次
  • 性别: Icon_minigender_1
  • 来自: 苏州
社区版块
存档分类
最新评论

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

 
阅读更多
转载:http://www.zeali.net/entry/70 MaDe1nZEAL
Http客户端程序已集成在Java语言中,可以通过URLConnection类调用。遗憾的是,由于SUN没有公布Http客户程序的源码,它实现的细节仍是一个谜。本文根据HTTP协议规范,用Java.net.Socket类实现一个HTTP协议客户端程序.

view plainprint?

    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; 
     
        } 
     
    } 

PS. 在这里可以找到Http Server端的最简单实现版本。
分享到:
评论

相关推荐

    java实现一个邮件客户端

    1. **JavaMail API**: JavaMail API是Java平台的核心组件之一,提供了处理SMTP(简单邮件传输协议)、POP3(邮局协议)和IMAP(因特网消息访问协议)的接口和类。它允许开发者创建、读取、发送和管理电子邮件。 2. ...

    java http客户端

    源程序代码将提供一个简单的 HTTP 客户端的实现代码,使用 Java 语言编写,能够根据给定的 URL,获得 URL 指向的资源,并对获得的内容进行进一步的解析和处理。 八、课程设计总结 通过本实验,使学生掌握了网络...

    ftp客户端源码 java实现

    本项目名为“ftp客户端源码 java实现”,提供了一个简单易用的FTP客户端示例,适用于学习和开发用途。 FTP客户端的基本功能包括连接到FTP服务器、登录、上传文件、下载文件、列出目录内容、创建或删除远程目录等。...

    java实现的opc ua 客户端/服务端的简单例子

    这个“java实现的opc ua客户端/服务端的简单例子”是一个很好的起点,对于初学者来说,可以帮助理解 OPC UA 的核心概念和Java实现。 首先,OPC UA 包含了客户端和服务端两部分,客户端用于请求和访问服务器上的数据...

    Java实现邮件客户端

    本文将深入探讨如何使用Java来实现一个邮件系统客户端,结合提供的标题和描述,我们将重点关注以下几个核心知识点: 1. **JavaMail API**:JavaMail API是Java平台上用于处理邮件的一组接口和类,它提供了发送、...

    http协议客户端编程

    ### HTTP协议客户端编程 #### 一、概述 HTTP(HyperText Transfer Protocol)是互联网上应用最为广泛的一种网络协议,主要用于从WWW服务器传输超文本到本地浏览器的传输协议。本文主要介绍如何利用Java语言中的`...

    coap 协议实现(服务端+客户端)

    本项目提供了COAP协议的服务器端和客户端的Java实现,使得开发者能够快速构建基于COAP的物联网应用。 **COAP协议简介** COAP是基于UDP的协议,它借鉴了HTTP的模型,具有RESTful架构,支持GET、PUT、POST和DELETE等...

    交通部809协议JAVA实现的接收车辆GPS数据的服务端代码

    本项目使用JAVA + MINA + ServiceLoader实现交通部809协议服务端代码。代码中完成了主链路部分,包括对客户端登录验证及应答,注销及应答,保持连接及应答,接收实时定位数据及历史定位数据等。如果扩充新业务只需要...

    Java语言实现Telnet客户端程序(有界面)

    Java语言实现的Telnet客户端程序是一种能够连接到远程服务器并执行命令或进行交互的工具。在本资源中,开发者提供了一个具有图形用户界面(GUI)的实现,使得用户可以通过直观的界面来操作 Telnet 客户端,而不再...

    JAVA实现FTP协议中的服务器和客户端\\

    本文将深入探讨如何使用Java实现FTP协议中的服务器和客户端。通过学习这些知识点,开发者可以构建自己的FTP应用,进行文件的上传、下载和管理。 首先,我们要了解FTP协议的基本工作原理。FTP基于TCP/IP模型,通过...

    telnet.rar telnet 协议 客户端 源代码 java

    本文将深入探讨telnet协议的工作原理,并重点分析Java实现telnet客户端的源代码,帮助读者深化对网络编程的理解。 一、telnet协议概述 telnet协议是一种基于TCP/IP的应用层协议,其主要功能是提供一个标准的接口,...

    Http服务器与客户端实例(Java版)

    在这个Java版的HTTP服务器与客户端实例中,我们将探讨如何利用Java来实现HTTP服务器的基本功能,并创建一个简单的客户端进行通信。 一、HTTP服务器 在Java中,我们可以使用内置的`java.net`和`java.io`库来构建一...

    电子邮件客户端_Java实现

    电子邮件客户端的实现是Java编程中的一个重要应用场景,它允许用户通过编程方式发送、接收和管理电子邮件。本项目提供了完整的源代码和可运行程序,方便开发者学习和使用。 在Java中实现电子邮件客户端,主要涉及到...

    java socket 客户端代码

    下面是一个简单的Java Socket客户端示例,用于连接到一个聊天服务器: ```java import java.io.*; import java.net.*; public class ChatClient { public static void main(String[] args) { try (Socket socket...

    JAVA开发邮件客户端

    在JAVA开发邮件客户端的过程中,涉及的关键技术和知识点相当广泛,涵盖了网络通信、邮件协议以及JAVA编程等多个领域。以下是对这些核心内容的详细解析: 1. **SMTP协议**:Simple Mail Transfer Protocol,简单邮件...

    kmip客户端简单实现

    KMIP4J是一个Java库,它是对KMIP协议的实现,允许开发者在Java应用中集成KMIP服务。该库提供了丰富的API,用于执行各种KMIP操作,如创建、检索、更新和销毁密钥。在本例中,我们将使用KMIP4J库创建一个简单的KMIP...

    UDP简易客户端与服务器端程序(Java)

    在这个Java实现的UDP简易客户端与服务器端程序中,我们可以通过两个核心文件——UDPClient.java和UDPServer.java来理解其工作原理。 首先,让我们来看看`UDPServer.java`。服务器端通常负责监听特定的端口,接收...

    Java基于socket实现的客户端和服务端通信功能完整实例

    Java基于socket实现的客户端和服务端通信功能完整实例 在Java中,Socket是实现网络通信的基础,通过Socket可以实现客户端和服务器端之间的通信。本文将详细介绍Java基于Socket实现的客户端和服务端通信功能,包括...

    服务端和客户端,java,C++实现代码

    Java以其跨平台的特性而闻名,它的Socket编程接口使得实现服务端和客户端通信变得相对简单。在Java中,服务端通常创建一个ServerSocket监听特定的端口,等待客户端的连接请求。一旦连接建立,服务器就可以通过Socket...

    java应用netty服务端和客户端

    在"java应用netty服务端和客户端"的示例中,Netty被用来构建一个简单的通信系统,其中包含服务端(Server)和客户端(Client)。为了实现通信,服务端和客户端都需要定义自己的`model对象`,这些对象通常包含了数据...

Global site tag (gtag.js) - Google Analytics