[转]客户端协议处理框架(URLConnection)

转自:http://blog.csdn.net/pipisky2006/article/details/6321411

 

客户端在与原车服务器通信时,都需要建立与远程服务器的连接,然后发送和接收与协议相符的数据。客户程序还需要对服务器发送的数据进行处理,有时把它们转换成相应的Java对象。Java对客户程序的通信过程进行了抽象,提供了通用的协议处理框架。这个框架封装了Socket,主要包含以下的几个类:

URL,统一资源定位器(Uniform Resource Locator),表示客户程序要访问的远程资源。

URLConnection,表示客户程序与远程服务器的连接。客户程序可以从该类中获得数据的输入流 和输出流。

URLStreamHandler,协议处理器,主要负责创建与协议相关的URLConnection对象。

ConentHandler,内容处理器,负责解析服务器发送的数据,把它转换为相应的Java对象。

以上类都位于java.net包中,除了URL类为具体类以外,其余的三个类都是抽象类,对于一种具体的协议,需要创建相应的URLConnection、URLStreamHandler和ContentHandler具体子类。

URLStreamHandlerFactory是工厂类,它的createURLStreamHandler()方法负责构造与特定协议相关的URLStreamHandler子类的实例。
ContentHandlerFactory也是工厂类,它的createContentHandler()方法负责构造与特定协议相关的ContentHandler子类的实例。
URLStreamHandler的openConnection()方法负责构造与特定协议相关的URLConnection子类的实例。

 

客户端协议处理框架的应用有两种,

1,运用协议处理框架的客户程序:指的是网络应用层的客户端程序。在客户端程序中,一般只需要访问框架的URL类和URLConnection类。其余的类或接口对客户程序都是透明的。

2,协议处理框架的实现程序:根据特定的协议,来扩展框架,创建URLConnection,URLStreamHandler和ConentHandler的具体子类,并且实现URLStreamHandlerFactory和ContentHandlerFactory接口。

 

SUN公司为协议处理框架提供了基于HTTP协议的实现,不过,这些实现类并没有在JDK类库中公开,它们都位于sun.net.www包或者其子包中。

 

在通过URL的构造方法URL(String url)创建一个URL对象时,构造方法会根据参数url中的协议符号,来创建一个与协议匹配的URLStreamHandler实例。在本例中,协议符号为“http”。URL的构造方法创建URLStreamHandler实例的流程如下。

1,如果在URL缓存中已经存在这样的URLStreamHandler实例,则无需在创建新的URLStreamHandler实例。否则继续执行下一步

2,如果程序已经通过URL类的静态setURLStreamHandlerFactory()方法设置了URLStreamHandlerFactory接口的具体实现类,那么就通过这个工厂类createURLStreamHandler()方法来构造一个URLStreamHandler实例。否则继续执行下一步。

3,根据系统属性java.protocol.handler.pkgs来决定URLStreamHandler具体子类的名字,然后对其实例化。例如:

java -Djava.protocol.handler.pkgs = com.abc.net.www | org.javathinker.protocols.HttpClient1

-D选项是用来设置系统属性。

如果这些属性中还是查找不到,那么继续执行下一步

4,试图实例化位于sun.net.www.protocol包中的sun.net.www.protocol.协议名.Handler类,如果失败,抛出MalformedURLException.

 

URL类的openConnection()方法创建并返回一个URLConnection对象,这个openConnection()方法实际上是通过调用URLStreamHandler类的openConnection()方法,来创建URLConnection对象的。
URL类的openStream()方法返回用于读取服务器发送数据的输入流,该方法实际上通过调用URLConnection类的getInputStream()方法来获得输入流。
URL类的getContent()方法返回包装了服务器发送数据的Java对象,该方法实际上调用URLConnection类的getContent()方法,而URLConnection类的getContent()方法又调用了ContentHandler类的getContent()方法。

 

URLConnection类表示客户程序与远程服务器的连接。
URLConnection有两个boolean类型的属性以及相应的get和set方法:
doInput属性:如果取值为true,表示允许获得输入流,读取远程服务器发送的数据。该属性的默认值为true。程序可通过getDoInput()和setDoInput()方法来读取和设置该属性。
doOutput属性:如果取值为true,表示允许获得输出流,向远程服务器发送数据。该属性的默认值为false。程序可通过getDoOutput()和setDoOutput()方法来读取和设置该属性。

URLConnection类提供了读取远程服务器的响应数据的一系列方法:
getHeaderField(String name):返回响应头中参数name指定的属性的值。
getContentType():返回响应正文的类型。如果无法获取响应正文的类型,就返回null。对于HTTP响应结果,在响应头中可能会包含响应正文的类型信息。
getContentLength():返回响应正文的长度。如果无法获取响应正文的长度,就返回-1。对于HTTP响应结果,在响应头中可能会包含响应正文的长度信息。
getContentEncoding():返回响应正文的编码类型。如果无法获取响应正文的编码类型,就返回null。对于HTTP响应结果,在响应头中可能会包含响应正文的编码类型信息。

 

客户程序可以采用以下几种方式来判断响应正文的类型:
(1)调用URLConnection类的getContentType()方法。
(2)调用URLConnection类的静态guessContentTypeFromName(String fname)方法,参数fname表示URL地址中的文件名部分。
(3)调用URLConnection类的静态guessContentTypeFromStream (InputStream in)方法,参数in表示输入流。

 

下面来看一些实例,

 

[java]  view plain  copy
 
  1. public class HttpInvoker {  
  2.   
  3.     public static final String GET_URL = "http://localhost:8080/welcome1";  
  4.   
  5.     public static final String POST_URL = "http://localhost:8080/welcome1";  
  6.   
  7.     public static void readContentFromGet() throws IOException {  
  8.         // 拼凑get请求的URL字串,使用URLEncoder.encode对特殊和不可见字符进行编码  
  9.         String getURL = GET_URL + "?username="  
  10.                 + URLEncoder.encode("fat man""utf-8");  
  11.         URL getUrl = new URL(getURL);  
  12.         // 根据拼凑的URL,打开连接,URL.openConnection函数会根据URL的类型,  
  13.         // 返回不同的URLConnection子类的对象,这里URL是一个http,因此实际返回的是HttpURLConnection  
  14.         HttpURLConnection connection = (HttpURLConnection) getUrl  
  15.                 .openConnection();  
  16.         // 进行连接,但是实际上get request要在下一句的connection.getInputStream()函数中才会真正发到  
  17.         // 服务器  
  18.         connection.connect();  
  19.         // 取得输入流,并使用Reader读取  
  20.         BufferedReader reader = new BufferedReader(new InputStreamReader(  
  21.                 connection.getInputStream()));  
  22.         System.out.println("=============================");  
  23.         System.out.println("Contents of get request");  
  24.         System.out.println("=============================");  
  25.         String lines;  
  26.         while ((lines = reader.readLine()) != null) {  
  27.             System.out.println(lines);  
  28.         }  
  29.         reader.close();  
  30.         // 断开连接  
  31.         connection.disconnect();  
  32.         System.out.println("=============================");  
  33.         System.out.println("Contents of get request ends");  
  34.         System.out.println("=============================");  
  35.     }  
  36.   
  37.     public static void readContentFromPost() throws IOException {  
  38.         // Post请求的url,与get不同的是不需要带参数  
  39.         URL postUrl = new URL(POST_URL);  
  40.         // 打开连接  
  41.         HttpURLConnection connection = (HttpURLConnection) postUrl  
  42.                 .openConnection();  
  43.         // Output to the connection. Default is  
  44.         // false, set to true because post  
  45.         // method must write something to the  
  46.         // connection  
  47.         // 设置是否向connection输出,因为这个是post请求,参数要放在  
  48.         // http正文内,因此需要设为true  
  49.         connection.setDoOutput(true);  
  50.         // Read from the connection. Default is true.  
  51.         connection.setDoInput(true);  
  52.         // Set the post method. Default is GET  
  53.         connection.setRequestMethod("POST");  
  54.         // Post cannot use caches  
  55.         // Post 请求不能使用缓存  
  56.         connection.setUseCaches(false);  
  57.         // This method takes effects to  
  58.         // every instances of this class.  
  59.         // URLConnection.setFollowRedirects是static函数,作用于所有的URLConnection对象。  
  60.         // connection.setFollowRedirects(true);  
  61.   
  62.         // This methods only  
  63.         // takes effacts to this  
  64.         // instance.  
  65.         // URLConnection.setInstanceFollowRedirects是成员函数,仅作用于当前函数  
  66.         connection.setInstanceFollowRedirects(true);  
  67.         // Set the content type to urlencoded,  
  68.         // because we will write  
  69.         // some URL-encoded content to the  
  70.         // connection. Settings above must be set before connect!  
  71.         // 配置本次连接的Content-type,配置为application/x-www-form-urlencoded的  
  72.         // 意思是正文是urlencoded编码过的form参数,下面我们可以看到我们对正文内容使用URLEncoder.encode  
  73.         // 进行编码  
  74.         connection.setRequestProperty("Content-Type",  
  75.                 "application/x-www-form-urlencoded");  
  76.         // 连接,从postUrl.openConnection()至此的配置必须要在connect之前完成,  
  77.         // 要注意的是connection.getOutputStream会隐含的进行connect。  
  78.         connection.connect();  
  79.         DataOutputStream out = new DataOutputStream(connection  
  80.                 .getOutputStream());  
  81.         // The URL-encoded contend  
  82.         // 正文,正文内容其实跟get的URL中'?'后的参数字符串一致  
  83.         String content = "firstname=" + URLEncoder.encode("一个大肥人""utf-8");  
  84.         // DataOutputStream.writeBytes将字符串中的16位的unicode字符以8位的字符形式写道流里面  
  85.         out.writeBytes(content);   
  86.   
  87.         out.flush();  
  88.         out.close(); // flush and close  
  89.         BufferedReader reader = new BufferedReader(new InputStreamReader(  
  90.                 connection.getInputStream()));  
  91.         String line;  
  92.         System.out.println("=============================");  
  93.         System.out.println("Contents of post request");  
  94.         System.out.println("=============================");  
  95.         while ((line = reader.readLine()) != null) {  
  96.             System.out.println(line);  
  97.         }  
  98.         System.out.println("=============================");  
  99.         System.out.println("Contents of post request ends");  
  100.         System.out.println("=============================");  
  101.         reader.close();  
  102.         connection.disconnect();  
  103.     }  
  104.   
  105.     /** 
  106.      * @param args 
  107.      */  
  108.     public static void main(String[] args) {  
  109.         // TODO Auto-generated method stub  
  110.         try {  
  111.             readContentFromGet();  
  112.             readContentFromPost();  
  113.         } catch (IOException e) {  
  114.             // TODO Auto-generated catch block  
  115.             e.printStackTrace();  
  116.         }  
  117.     }  
  118.   
  119. }  


上文的代码是参考,http://blog.csdn.net/pandazxx/article/details/1657109,在异常处理和错误重连等机制并未处理的太好,不妨碍我们学习主要内容。

 

1,encode,在get的url和post的发送的内容content,需要进行URLEncoder。

2,HttpURLConnection的connect()函数,实际上只是建立了一个与服务器的tcp连接,并没有实际发送http请求。
3, 无论是post还是get,http请求实际上直到HttpURLConnection的getInputStream()这个函数里面才正式发送出去。
4, 在用POST方式发送URL请求时,URL请求参数的设定顺序是重中之重, 对connection对象的一切配置(那一堆set函数) 都必须要在connect()函数执行之前完成。
5,http请求实际上由两部分组成,一个是http头,所有关于此次http请求的配置都在http头里面定义,一个是正文content。connect()函数会根据HttpURLConnection对象的配置值生成http头部信息,因此在调用connect函数之前,

6,注意  connection.disconnect(),断开链接时应该调用,而且应该在finally中进行,上文的exception并未处理的太好。

7,connection.setConnectTimeout(CONNECT_TIMEOUT);
connection.setReadTimeout(READ_TIMEOUT);

注意设置连接和读入的超时时间,对于移动互联网来说,30s足够了。

8,使用connection.setRequestProperty(),添加自己所需的恰当的属性,例如断点续传,.setRequestProperty("RANGE","bytes=102400-");在cmwap中一定要注意的。

 

如果链接可以打开的话,可以看看这个,更全面些,http://developer.Android.com/reference/java/net/HttpURLConnection.html

猜你喜欢

转载自currentj.iteye.com/blog/2296059