HttpURLConnection学习

最经常使用的Http请求无非是get和post,get请求能够获取静态页面,也能够把参数放在URL字串后面,传递给servlet,post与get的不一样之处在于post的参数不是放在URL字串里面,而是放在http请求的正文内。
在Java中可使用HttpURLConnection发起这两种请求,了解此类,对于了解soap,和编写servlet的自动测试代码都有很大的帮助。
下面的代码简单描述了如何使用HttpURLConnection发起这两种请求,以及传递参数的方法:
缓存

public   class  HttpInvoker {

    
public   static   final  String GET_URL  =   " http://localhost:8080/welcome1 " ;

    
public   static   final  String POST_URL  =   " http://localhost:8080/welcome1 " ;

    
public   static   void  readContentFromGet()  throws  IOException {
        
//  拼凑get请求的URL字串,使用URLEncoder.encode对特殊和不可见字符进行编码
        String getURL  =  GET_URL  +   " ?username= "
                
+  URLEncoder.encode( " fat man " " utf-8 " );
        URL getUrl 
=   new  URL(getURL);
        
//  根据拼凑的URL,打开链接,URL.openConnection函数会根据URL的类型,
        
//  返回不一样的URLConnection子类的对象,这里URL是一个http,所以实际返回的是HttpURLConnection
        HttpURLConnection connection  =  (HttpURLConnection) getUrl
                .openConnection();
        
//  进行链接,可是实际上get request要在下一句的connection.getInputStream()函数中才会真正发到
        
//  服务器
        connection.connect();
        
//  取得输入流,并使用Reader读取
        BufferedReader reader  =   new  BufferedReader( new  InputStreamReader(
                connection.getInputStream()));
        System.out.println(
" ============================= " );
        System.out.println(
" Contents of get request " );
        System.out.println(
" ============================= " );
        String lines;
        
while  ((lines  =  reader.readLine())  !=   null {
            System.out.println(lines);
        }
        reader.close();
        
//  断开链接
        connection.disconnect();
        System.out.println(
" ============================= " );
        System.out.println(
" Contents of get request ends " );
        System.out.println(
" ============================= " );
    }

    
public   static   void  readContentFromPost()  throws  IOException {
        
//  Post请求的url,与get不一样的是不须要带参数
        URL postUrl  =   new  URL(POST_URL);
        
//  打开链接
        HttpURLConnection connection  =  (HttpURLConnection) postUrl
                .openConnection();
        
//  Output to the connection. Default is
        
//  false, set to true because post
        
//  method must write something to the
        
//  connection
        
//  设置是否向connection输出,由于这个是post请求,参数要放在
        
//  http正文内,所以须要设为true
        connection.setDoOutput( true );
        
//  Read from the connection. Default is true.
        connection.setDoInput( true );
        
//  Set the post method. Default is GET
        connection.setRequestMethod( " POST " );
        
//  Post cannot use caches
        
//  Post 请求不能使用缓存
        connection.setUseCaches( false );
        
//  This method takes effects to
        
//  every instances of this class.
        
//  URLConnection.setFollowRedirects是static函数,做用于全部的URLConnection对象。
        
//  connection.setFollowRedirects(true);

        
//  This methods only
        
//  takes effacts to this
        
//  instance.
        
//  URLConnection.setInstanceFollowRedirects是成员函数,仅做用于当前函数
        connection.setInstanceFollowRedirects( true );
        
//  Set the content type to urlencoded,
        
//  because we will write
        
//  some URL-encoded content to the
        
//  connection. Settings above must be set before connect!
        
//  配置本次链接的Content-type,配置为application/x-www-form-urlencoded的
        
//  意思是正文是urlencoded编码过的form参数,下面咱们能够看到咱们对正文内容使用URLEncoder.encode
        
//  进行编码
        connection.setRequestProperty( " Content-Type " ,
                
" application/x-www-form-urlencoded " );
        
//  链接,从postUrl.openConnection()至此的配置必需要在connect以前完成,
        
//  要注意的是connection.getOutputStream会隐含的进行connect。
        connection.connect();
        DataOutputStream out 
=   new  DataOutputStream(connection
                .getOutputStream());
        
//  The URL-encoded contend
        
//  正文,正文内容其实跟get的URL中'?'后的参数字符串一致
        String content  =   " firstname= "   +  URLEncoder.encode( " 一个大肥人 " " utf-8 " );
        
//  DataOutputStream.writeBytes将字符串中的16位的unicode字符以8位的字符形式写道流里面
        out.writeBytes(content); 

        out.flush();
        out.close(); 
//  flush and close
        BufferedReader reader  =   new  BufferedReader( new  InputStreamReader(
                connection.getInputStream()));
        String line;
        System.out.println(
" ============================= " );
        System.out.println(
" Contents of post request " );
        System.out.println(
" ============================= " );
        
while  ((line  =  reader.readLine())  !=   null {
            System.out.println(line);
        }
        System.out.println(
" ============================= " );
        System.out.println(
" Contents of post request ends " );
        System.out.println(
" ============================= " );
        reader.close();
        connection.disconnect();
    }

    
/**   *//**
     * 
@param  args
     
*/
    
public   static   void  main(String[] args) {
        
//  TODO Auto-generated method stub
         try  {
            readContentFromGet();
            readContentFromPost();
        } 
catch  (IOException e) {
            
//  TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

}


上面的readContentFromGet()函数产生了一个get请求,传给servlet一个username参数,值为"fat man"。
readContentFromPost()函数产生了一个post请求,传给servlet一个firstname参数,值为"一个大肥人"。
HttpURLConnection.connect函数,实际上只是创建了一个与服务器的tcp链接,并无实际发送http请求。不管是post仍是get,http请求实际上直到HttpURLConnection.getInputStream()这个函数里面才正式发送出去。

readContentFromPost() 中,顺序是重中之重,对connection对象的一切配置(那一堆set函数)都必需要在connect()函数执行以前完成。而对 outputStream的写操做,又必需要在inputStream的读操做以前。这些顺序其实是由http请求的格式决定的。

http 请求实际上由两部分组成,一个是http头,全部关于这次http请求的配置都在http头里面定义,一个是正文content,在connect()函数里面,会根据HttpURLConnection对象的配置值生成http头,所以在调用connect函数以前,就必须把全部的配置准备好。

紧接着http头的是http请求的正文,正文的内容经过outputStream写入,实际上outputStream不是一个网络流,充其量是个字符串流,往里面写入的东西不会当即发送到网络,而是在流关闭后,根据输入的内容生成http正文。

至此,http请求的东西已经准备就绪。在getInputStream()函数调用的时候,就会把准备好的http请求正式发送到服务器了,而后返回一个输入流,用于读取服务器对于这次http请求的返回信息。因为http请求在getInputStream的时候已经发送出去了(包括http头和正文),所以在getInputStream()函数以后对connection对象进行设置(对http头的信息进行修改)或者写入 outputStream(对正文进行修改)都是没有意义的了,执行这些操做会致使异常的发生
上节说道,post请求的OutputStream实际上不是网络流,而是写入内存,在getInputStream中才真正把写道流里面的内容做为正文与根据以前的配置生成的http request头合并成真正的http request,并在此时才真正向服务器发送。

HttpURLConnection.setChunkedStreamingMode 函数能够改变这个模式,设置了ChunkedStreamingMode后,再也不等待OutputStream关闭后生成完整的http request一次过发送,而是先发送http request头,正文内容则是网路流的方式实时传送到服务器。其实是不告诉服务器http正文的长度,这种模式适用于向服务器传送较大的或者是不容易获取长度的数据,如文件。
服务器

public   static   void  readContentFromChunkedPost()  throws  IOException {
        URL postUrl 
=   new  URL(POST_URL);
        HttpURLConnection connection 
=  (HttpURLConnection) postUrl
                .openConnection();
        connection.setDoOutput(
true );
        connection.setDoInput(
true );
        connection.setRequestMethod(
" POST " );
        connection.setUseCaches(
false );
        connection.setInstanceFollowRedirects(
true );
        connection.setRequestProperty(
" Content-Type " ,
                
" application/x-www-form-urlencoded " );
        
/** //*
         * 与readContentFromPost()最大的不一样,设置了块大小为5字节
         
*/
        connection.setChunkedStreamingMode(
5 );
        connection.connect();
        
/** //*
         * 注意,下面的getOutputStream函数工做方式于在readContentFromPost()里面的不一样
         * 在readContentFromPost()里面该函数仍在准备http request,没有向服务器发送任何数据
         * 而在这里因为设置了ChunkedStreamingMode,getOutputStream函数会根据connect以前的配置
         * 生成http request头,先发送到服务器。
         
*/
        DataOutputStream out 
=   new  DataOutputStream(connection
                .getOutputStream());
        String content 
=   " firstname= "   +  URLEncoder.encode( " 一个大肥人                                                                                "   +
                
"                                            "   +
                
" asdfasfdasfasdfaasdfasdfasdfdasfs " " utf-8 " );
        out.writeBytes(content); 

        out.flush();
        out.close(); 
//  到此时服务器已经收到了完整的http request了,而在readContentFromPost()函数里,要等到下一句服务器才能收到http请求。
        BufferedReader reader  =   new  BufferedReader( new  InputStreamReader(
                connection.getInputStream()));
        
        out.flush();
        out.close(); 
//  flush and close
        String line;
        System.out.println(
" ============================= " );
        System.out.println(
" Contents of post request " );
        System.out.println(
" ============================= " );
        
while  ((line  =  reader.readLine())  !=   null {
            System.out.println(line);
        }
        System.out.println(
" ============================= " );
        System.out.println(
" Contents of post request ends " );
        System.out.println(
" ============================= " );        reader.close();        connection.disconnect();    }
相关文章
相关标签/搜索