转自:http://mobile.51cto.com/abased-448264.htm

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

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

上面的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正文的长度,这种模式适用于向服务器传送较大的或者是不容易获取长度的数据,如文件。

    1. public   static   void  readContentFromChunkedPost()  throws  IOException {
    2. URL postUrl  =   new  URL(POST_URL);
    3. HttpURLConnection connection  =  (HttpURLConnection) postUrl
    4. .openConnection();
    5. connection.setDoOutput( true );
    6. connection.setDoInput( true );
    7. connection.setRequestMethod( " POST " );
    8. connection.setUseCaches( false );
    9. connection.setInstanceFollowRedirects( true );
    10. connection.setRequestProperty( " Content-Type " ,
    11. " application/x-www-form-urlencoded " );
    12. /** //*
    13. * 与readContentFromPost()最大的不同,设置了块大小为5字节
    14. */
    15. connection.setChunkedStreamingMode( 5 ); //设置该值后,如果网络中断,会自动重新请求,如果不想自动重新请求,可设置为0
    16. connection.connect();
    17. /** //*
    18. * 注意,下面的getOutputStream函数工作方式于在readContentFromPost()里面的不同
    19. * 在readContentFromPost()里面该函数仍在准备http request,没有向服务器发送任何数据
    20. * 而在这里由于设置了ChunkedStreamingMode,getOutputStream函数会根据connect之前的配置
    21. * 生成http request头,先发送到服务器。
    22. */
    23. DataOutputStream out  =   new  DataOutputStream(connection
    24. .getOutputStream());
    25. String content  =   " firstname= "   +  URLEncoder.encode( " 一个大肥人                                                                                "   +
    26. "                                            "   +
    27. " asdfasfdasfasdfaasdfasdfasdfdasfs " ,  " utf-8 " );
    28. out.writeBytes(content);
    29. out.flush();
    30. out.close();  //  到此时服务器已经收到了完整的http request了,而在readContentFromPost()函数里,要等到下一句服务器才能收到http请求。
    31. BufferedReader reader  =   new  BufferedReader( new  InputStreamReader(
    32. connection.getInputStream()));
    33. out.flush();
    34. out.close();  //  flush and close
    35. String line;
    36. System.out.println( " ============================= " );
    37. System.out.println( " Contents of post request " );
    38. System.out.println( " ============================= " );
    39. while  ((line  =  reader.readLine())  !=   null ) {
    40. System.out.println(line);
    41. }
    42. System.out.println( " ============================= " );
    43. System.out.println( " Contents of post request ends " );
    44. System.out.println( " ============================= " );
    45. reader.close();
    46. connection.disconnect();
    47. }

HttpURLConnection学习的更多相关文章

  1. java调用Http请求 -HttpURLConnection学习

    最常用的Http请求无非是get和post,get请求可以获取静态页面,也可以把参数放在URL字串后面,传递给servlet,post与get的不同之处在于post的参数不是放在URL字串里面,而是放 ...

  2. Android上传图片(PHP服务器)

    原理 Android客户端模拟一个HTTP的Post请求到服务器端,服务器端接收相应的Post请求后,返回响应信息给给客户端. PHP服务器 <?php move_uploaded_file($ ...

  3. Http学习之使用HttpURLConnection发送post和get请求(2)

    接上节Http学习之使用HttpURLConnection发送post和get请求 本节深入学习post请求. 上 节说道,post请求的OutputStream实际上不是网络流,而是写入内存,在ge ...

  4. android菜鸟学习笔记24----与服务器端交互(一)使用HttpURLConnection和HttpClient请求服务端数据

    主要是基于HTTP协议与服务端进行交互. 涉及到的类和接口有:URL.HttpURLConnection.HttpClient等 URL: 使用一个String类型的url构造一个URL对象,如: U ...

  5. 【转】 Pro Android学习笔记(七二):HTTP服务(6):HttpURLConnection

    目录(?)[-] Http Get的使用方式 基础小例子 Cookie的使用 重定向 HTTP POST的小例子 基础小例子 文章转载只能用于非商业性质,且不能带有虚拟货币.积分.注册等附加条件,转载 ...

  6. Http学习之使用HttpURLConnection发送post和get请求(3)

    使用HttpURLConnection发送post和get请求 但我们常常会碰到这样一种情况: 通过HttpURLConnection来模拟模拟用户登录Web服务器,服务器使用cookie进行用户认证 ...

  7. Http学习之使用HttpURLConnection发送post和get请求(1)

    最常用的Http请求无非是get和post,get请求可以获取静态页面,也可以把参数放在URL字串后面,传递给servlet,post与get的不同之处在于post的参数不是放在URL字串里面,而是放 ...

  8. OKHttp源码学习--HttpURLConnection HttpClient OKHttp Get and post Demo用法对比

    1.HttpURLConnection public class HttpURLConnectionGetAndPost { private String urlAddress = "xxx ...

  9. HttpURLConnection与HttpClient学习

    转载HttpURLConnection与HttpClient浅析 一.HttpURLConnection的使用 import org.slf4j.Logger; import org.slf4j.Lo ...

随机推荐

  1. C# string转int

    1,int转成string用toString 或者Convert.toString()如下 例如:int varInt = 1; string varString = Convert.ToString ...

  2. 在万网虚拟主机上部署MVC5

    参考 要想部署mvc,需要把一些mvc用到的全局程序集改为本地部署,通过N次试验,终于搞定. 特写个备忘录,免得以后忘了. 首先更改web.config,在里面加上 <system.web> ...

  3. Flightgear 编译

    一.FlightGear简介 FlightGear 始于1997年,是一个开源的多平台飞行模拟器. 二.FlightGear编译过程 FlightGear平台的说明文档见:http://wiki.fl ...

  4. SGU 172.eXam(二分图染色)

    时间限制:0.25s 空间限制:4M 题意: 将n(n<200)个点分成两个集合,给出m(m<=30000)对不能在一个集合的点对,判断能否分成满足要求的集合,输出其中一个集合和集合的总数 ...

  5. SGU 195. New Year Bonus Grant

    时间限制:0.75s 空间限制:4M 题意: 在一颗树(最多500000个节点)中,可以对节点染色,但是一个节点染了色后,它的父节点和兄弟节点都不能再染了,求最大的染色节点数,并输出所有染色节点. S ...

  6. js作用域与作用域链

    一直对Js的作用域有点迷糊,今天偶然读到JavaScript权威指南,立马被吸引住了,写的真不错.我看的是第六版本,相当的厚,大概1000多页,Js博大精深,要熟悉精通需要大毅力大功夫. 一:函数作用 ...

  7. JQuery 点击控件获取当前坐标时不兼容IE7

    现在要求在点击文本框时,获取文本框的坐标,需要相对文本框的位置来显示信息. 思路就是,绑定文本框的click 事件,一旦有点击就触发,去调用clickevent 函数执行计算. $('#txt_m') ...

  8. Windows7 IIS7 无法启动计算机上的服务W3SVC如何修复

    错误提示 启动iis7管理服务器提示:无法启动计算机上的服务W3SVC 启动Windows Process Activation Service服务,报错:6801 指定资源管理器中的事务支持未启动或 ...

  9. 马的遍历问题-回溯法应用-ACM

    马的遍历问题 在n*m的棋盘中,马只能走“日” 字.马从位置(x,y)处出发,把棋盘的每一格都走一次,且只走一次.找出所有路径. 问题解的搜索空间? 棋盘的规模是n*m,是指行有n条边,列有m条边. ...

  10. ubuntu install opengrok

    总结: 1. 安装jdk和tomcat 2. 安装ctags 3. 解压opengrok.tar.gz包, 然后将source.war复制到tomcat/webapp下面 sudo cp -R ope ...