• [安卓] 14、安卓HTTP——POST和GET用法分析


    内容简介

    本文通过建立一个简单的Servlet服务器来分析安卓上用HTTP和服务器通信的细节,旨在演示C/S模式下服务器端和客户端的工作过程。

    目录

    part.1 用MyEclipse建立一个简单的servlet服务器

    part.2 安卓HTTP的POST和GET请求方法

    part.3 本例中C/S双方工作机制分析

    part.4 拓展知识


    注:这里首先假设您已经正确安装好了MyEclipse及Tomcat并做了相应的配置,可以支持开发并部署一个简单的Java Web工程;假设您已经安装了Eclipse并配置好Android相应开发环境。

    part.1 用MyEclipse建立一个简单的servlet服务器

    在MyEclipse中File->New->Other->Web Project->Next->Project Name取beautifulzzzz(随便)->Finish,从而新建一个Java Web Project。

    在web.xml中<welcome-file-list>标签对中指明了打开网站的首页为index.jsp,接着点击1号按钮选择一个服务器,然后点击2号按钮将web工程部署到该服务器上,然后在浏览器中输入http://localhost:8080/beautifulzzzz/就能看到相应的页面:

    现在在src中新建一个名为hello的servlet,并添加相应函数(最终如下):

      1 import java.io.IOException;
      2 import java.io.PrintWriter;
      3 import java.util.Map;
      4 
      5 import javax.servlet.ServletException;
      6 import javax.servlet.http.HttpServlet;
      7 import javax.servlet.http.HttpServletRequest;
      8 import javax.servlet.http.HttpServletResponse;
      9 
     10 public class hello extends HttpServlet {
     11 
     12     /**
     13      * Constructor of the object.
     14      */
     15     public hello() {
     16         super();
     17     }
     18 
     19     /**
     20      * Destruction of the servlet. <br>
     21      */
     22     public void destroy() {
     23         super.destroy(); // Just puts "destroy" string in log
     24         // Put your code here
     25     }
     26 
     27     /**
     28      * The doGet method of the servlet. <br>
     29      * 
     30      * This method is called when a form has its tag value method equals to get.
     31      * 
     32      * @param request
     33      *            the request send by the client to the server
     34      * @param response
     35      *            the response send by the server to the client
     36      * @throws ServletException
     37      *             if an error occurred
     38      * @throws IOException
     39      *             if an error occurred
     40      */
     41     /*
     42      * 以Get方式访问页面时执行该函数 执行doGet前会先执行getLastModified,如果浏览器发现getLastModified返回数值
     43      * 与上次访问返回数值相同,则认为该文档没有更新,浏览器执行缓存而不执行doGet 如果返回-1则认为是实时更新的,总是执行该函数
     44      */
     45     public void doGet(HttpServletRequest request, HttpServletResponse response)
     46             throws ServletException, IOException {
     47         this.log("执行 doGet 方法...");
     48         this.execute(request, response);
     49     }
     50 
     51     /**
     52      * The doPost method of the servlet. <br>
     53      * 
     54      * This method is called when a form has its tag value method equals to
     55      * post.
     56      * 
     57      * @param request
     58      *            the request send by the client to the server
     59      * @param response
     60      *            the response send by the server to the client
     61      * @throws ServletException
     62      *             if an error occurred
     63      * @throws IOException
     64      *             if an error occurred 执行前不会执行getLastModified
     65      */
     66     public void doPost(HttpServletRequest request, HttpServletResponse response)
     67             throws ServletException, IOException {
     68         this.log("执行 doPost 方法...");
     69         this.execute(request, response);
     70     }
     71 
     72     /**
     73      * 返回该Servlet生成文档的更新时间。对Get方法有效 返回的时间为相对于1970年1月1日08:00:00的毫秒数
     74      * 如果返回-1表示实时更新。默认为-1
     75      */
     76     @Override
     77     public long getLastModified(HttpServletRequest request) {
     78         this.log("执行 getLastModified 方法...");
     79         return -1;
     80     }
     81 
     82     // 执行方法
     83     private void execute(HttpServletRequest request,
     84             HttpServletResponse response) throws ServletException, IOException {
     85 
     86         response.setCharacterEncoding("UTF-8");// 设置request和response编码,两个都要注意
     87         request.setCharacterEncoding("UTF-8");
     88         String requestURI = request.getRequestURI();// 访问Servlet的URI
     89         String method = request.getMethod();// 访问Servlet的方式Get或Post
     90         // 获得用户提交的所有param
     91         Map<String, String> map = request.getParameterMap();
     92         for (String key : map.keySet()) {
     93             System.out.println(key + "+" + request.getParameter(key));
     94         }
     95 
     96         response.setContentType("text/html");// 设置文档类型为HTML类型
     97         PrintWriter out = response.getWriter();
     98         out.println("<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">");
     99         out.println("<HTML>");
    100         out.println("<meta http-equiv="content-type" content="text/html; charset=UTF-8">");
    101         out.println("<HEAD><TITLE>A Servlet</TITLE></HEAD>");
    102         out.println(" <BODY>");
    103         out.println("    以" + method + " 方式访问该页面。提取的param参数为:<br/>");
    104         for (String key : map.keySet()) {
    105             out.println("    " + key + "+" + request.getParameter(key) + "<br/>");
    106         }
    107 
    108         out.println("  </BODY>");
    109         out.println("</HTML>");
    110         out.flush();
    111         out.close();
    112     }
    113 
    114     /**
    115      * Initialization of the servlet. <br>
    116      * 
    117      * @throws ServletException
    118      *             if an error occurs
    119      */
    120     public void init() throws ServletException {
    121         // Put your code here
    122     }
    123 
    124 }

    这里将doGet和doPost都交给了execute执行,在execute中用request获取请求的相关信息,用response设置返回信息(这样如果用浏览器访问该网页时一般是HTTP的GET或POST请求将触发doGet或doPost函数,然后最终将请求提交给execute执行处理,在execute中获取请求信息并用response的response.getWriter()向客户端写回信息,这里由于是web服务,所以写回的是一个完整的html文档,这样浏览器就能根据返回的文档进行相应显示啦~)

    此外,当我们添加一个servlet时会发现web.xml中多了些东西:

    包括servlet的name和对应的class,特别重要的是下面的servlet-mapping中的url-pattern,这个指明了访问该servlet的地址:在这里为http://localhost:8080/beautifulzzzz/servlet/hello


    part.2 安卓HTTP的POST和GET请求方法

    摘自园友lingyun1120关于安卓HTTP的POST和GET的请求的总结:

    1.get是从服务器上获取数据,post是向服务器传送数据。
    2.get是把参数数据队列加到提交表单的 ACTION属性所指的URL中,值和表单内各个字段一一对应,在URL中可以看到。post是通过HTTPpost机制,将表单内各个字段与其内容放置 在HTML HEADER内一起传送到ACTION属性所指的URL地址。用户看不到这个过程。
    3.对于get方式,服务器端用 Request.QueryString获取变量的值,对于post方式,服务器端用Request.Form获取提交的数据。
    4.get 传送的数据量较小,不能大于2KB。post传送的数据量较大,一般被默认为不受限制。但理论上,IIS4中最大量为80KB,IIS5中为100KB。
    5.get安全性非常低,post安全性较高。

    对于安卓HTTP请求的实现主要有两种方法,一种是传统的HttpURLConnection 方式,另一种是HttpClinet方式。

    方式一:HttpURLConnection之GET

     1 /***
     2  * 用HttpURLConnection发送Get请求,返回请求字符
     3  * @return
     4  * @throws IOException
     5  */
     6 public String Func1() throws IOException{
     7     // 拼凑get请求的URL字串,使用URLEncoder.encode对特殊和不可见字符进行编码
     8     String MyURL=BASE_URL+ "?name=" + URLEncoder.encode("beautifulzzzz", "utf-8")
     9             +"&password=12345678";//(好像这里中文不行)
    10     URL getUrl = new URL(MyURL);
    11     // 根据拼凑的URL,打开连接,URL.openConnection函数会根据URL的类型,
    12     // 返回不同的URLConnection子类的对象,这里URL是一个http,因此实际返回的是HttpURLConnection
    13     HttpURLConnection conn = (HttpURLConnection) getUrl.openConnection();
    14     
    15     // 设置连接属性
    16     conn.setConnectTimeout(30000);// 设置连接超时时长,单位毫秒
    17           
    18     // 进行连接,但是实际上get request要在下一句的connection.getInputStream()函数中才会真正发到服务器
    19     BufferedReader reader = new BufferedReader(new InputStreamReader(
    20             conn.getInputStream()));// 取得输入流,并使用Reader读取
    21     String result = "";
    22     String line = "";
    23     while ((line = reader.readLine()) != null) {
    24         result = result + line+"
    ";
    25     }
    26     System.out.println(result);
    27     reader.close();
    28     conn.disconnect();
    29     return result;
    30 }

    因为Get请求请求的内容是放在URL中的,所以第8行用BASE_URL和想发送的键值对合成为新的URL,然后根据新合成的URL打开链接获得HttpURLConnection,但是真正的get请求是在connection.getInputStream()函数中才会真正发到服务器的,当该函数执行完时会返回一个输入流,然后我们使用Reader读取该输入流中的内容从而获得服务器的response。

    方式二:HttpURLConnection之POST

     1 /***
     2  * 用HttpURLConnection发送post请求,返回请求字符
     3  * @return
     4  * @throws IOException
     5  */
     6 public String Func2() throws IOException {
     7     URL url = new URL(BASE_URL);
     8     // 此处的urlConnection对象实际上是根据URL的
     9     // 请求协议(此处是http)生成的URLConnection类
    10     // 的子类HttpURLConnection,故此处最好将其转化
    11     // 为HttpURLConnection类型的对象,以便用到
    12     // HttpURLConnection更多的API.如下:
    13     HttpURLConnection conn = (HttpURLConnection) url.openConnection();
    14 
    15     // 设置连接属性
    16     conn.setDoOutput(true);// 使用 URL 连接进行输出
    17     conn.setDoInput(true);// 使用 URL 连接进行输入
    18     conn.setUseCaches(false);// POST请求不能用缓存
    19     conn.setConnectTimeout(30000);// 设置连接超时时长,单位毫秒
    20     conn.setInstanceFollowRedirects(true);// URLConnection.setInstanceFollowRedirects是成员函数,仅作用于当前函数
    21     // 配置本次连接的Content-type,配置为application/x-www-form-urlencoded的
    22     // 意思是正文是urlencoded编码过的form参数,下面我们可以看到我们对正文内容使用URLEncoder.encode
    23     // 进行编码
    24     conn.setRequestProperty("Content-Type",
    25             "application/x-www-form-urlencoded");
    26     conn.setRequestMethod("POST");// 设置请求方式,POST or
    27                                     // GET,注意:如果请求地址为一个servlet地址的话必须设置成POST方式
    28 
    29     OutputStream outStrm = conn.getOutputStream();// 此处getOutputStream会隐含的进行connect
    30     DataOutputStream out = new DataOutputStream(outStrm);
    31      // 正文,正文内容其实跟get的URL中'?'后的参数字符串一致
    32     String content = "name=" + URLEncoder.encode("李某人", "utf-8")
    33             +"&password="+ URLEncoder.encode("12345678", "utf-8");
    34     // DataOutputStream.writeBytes将字符串中的16位的unicode字符以8位的字符形式写道流里面
    35     out.writeBytes(content); 
    36     out.flush();
    37     out.close(); // flush and close
    38     
    39     // 调用HttpURLConnection连接对象的getInputStream()函数,
    40     // 将内存缓冲区中封装好的完整的HTTP请求电文发送到服务端。
    41     InputStream inStrm = conn.getInputStream(); // <===注意,实际发送请求的代码段就在这里
    42     // 上边的httpConn.getInputStream()方法已调用,本次HTTP请求已结束,再向对象输出流的输出已无意义,
    43     // 既使对象输出流没有调用close()方法,下边的操作也不会向对象输出流写入任何数据.
    44     // 因此,要重新发送数据时需要重新创建连接、重新设参数、重新创建流对象、重新写数据、
    45     // 重新发送数据(至于是否不用重新这些操作需要再研究)
    46     BufferedReader reader = new BufferedReader(
    47             new InputStreamReader(inStrm));
    48     String result = "";
    49     String line = "";
    50     while ((line = reader.readLine()) != null) {
    51         result = result + line+"
    ";
    52     }
    53     System.out.println(result);
    54     reader.close();
    55     conn.disconnect();
    56     return result;
    57 }

    对于POST请求和GET不同点在于POST的请求正文不是放在URL中。其信息包括请求头和请求正文,所有关于此次http请求的配置都在http头里面定义;对于请求正文content,在connect()函数里面,会根据HttpURLConnection对象的配置值生成http头,因此在调用connect函数之前,就必须把所有的配置准备好(但是如果使用了conn.getInputStream()函数就可以不用使用connect()函数了)。

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

    至此,http请求的东西已经准备就绪。在getInputStream()函数调用的时候,就会把准备好的http请求正式发送到服务器了,然后返回一 个输入流,用于读取服务器对于此次http请求的返回信息。由于http请求在getInputStream的时候已经发送出去了(包括http头和正 文),因此在getInputStream()函数之后对connection对象进行设置(对http头的信息进行修改)或者写入 outputStream(对正文进行修改)都是没有意义的了,执行这些操作会导致异常的发生。

    注:这里要注意24和35行,如果设置不对会导致服务器无法获取键值对!

    注:上面一段参考博客pandazxx的专栏:[Http学习之使用HttpURLConnection发送post和get请求 ]

    方式三:HttpClinet之GET

     1 /***
     2  * 使用Http的GET请求返回服务器返回结果字符串
     3  * 
     4  * @return
     5  * @throws ClientProtocolException
     6  * @throws IOException
     7  */
     8 public String Func3() throws ClientProtocolException, IOException {
     9     HttpGet httpGet = new HttpGet(BASE_URL + "?name=beautifulzzzz"
    10             + "&password=1234");
    11     // 获取HttpClient对象
    12     HttpClient httpClient = new DefaultHttpClient();
    13     // 连接超时
    14     httpClient.getParams().setParameter(
    15             CoreConnectionPNames.CONNECTION_TIMEOUT, 30000);
    16     // 请求超时
    17     httpClient.getParams().setParameter(CoreConnectionPNames.SO_TIMEOUT,
    18             30000);
    19     HttpResponse httpResp = httpClient.execute(httpGet);
    20     String response = EntityUtils.toString(httpResp.getEntity(), "UTF-8");
    21     System.out.println(response);
    22     if (response == null)
    23         response = "";
    24     return response;
    25 }

    当使用HttpClient发送Get请求时则相对简单,但是从第9行还是可以看出Get请求的消息还是放在URL中的,特别的这里是实例化一个HttpGet请求,并用HttpClient对象进行相关属性配置,然后调用execute函数获得服务器返回HttpResponse,然后调用getEntity()函数获取Httpresponse实体内容。

    方式四:HttpClinet之POST

     1 /***
     2  * 使用Http的POST请求返回服务器返回结果字符串
     3  * 
     4  * @return
     5  * @throws ClientProtocolException
     6  * @throws IOException
     7  */
     8 public String Func4() throws ClientProtocolException, IOException {
     9     // 將用戶名、密碼和imei封裝到list中,待http發送post請求給服務器
    10     NameValuePair pair1 = new BasicNameValuePair("user_name", "涛");
    11     NameValuePair pair2 = new BasicNameValuePair("user_password",
    12             "Deddd344");
    13     List<NameValuePair> pairList = new ArrayList<NameValuePair>();
    14     pairList.add(pair1);
    15     pairList.add(pair2);
    16     HttpPost httpPost = new HttpPost(BASE_URL);
    17     HttpEntity requestHttpEntity = new UrlEncodedFormEntity(pairList,
    18             HTTP.UTF_8);
    19     // 将请求体内容加入请求中
    20     httpPost.setEntity(requestHttpEntity);
    21     // 获取HttpClient对象
    22     HttpClient httpClient = new DefaultHttpClient();
    23     // 连接超时
    24     httpClient.getParams().setParameter(
    25             CoreConnectionPNames.CONNECTION_TIMEOUT, 30000);
    26     // 请求超时
    27     httpClient.getParams().setParameter(CoreConnectionPNames.SO_TIMEOUT,
    28             30000);
    29 
    30     HttpResponse httpResp = httpClient.execute(httpPost);
    31     String response = EntityUtils.toString(httpResp.getEntity(), "UTF-8");
    32     System.out.println(response);
    33     if (response == null)
    34         response = "";
    35     return response;
    36 }

    对于HttpClinet发送POST请求,因为键值不是保存在URL中,所以这里要用NameValuePair构建键值对,然后用List<NameValuePair>存储这些键值对,并用此List构建一个HttpEntity实体信息,然后调用httpPost.setEntity(requestHttpEntity);将实体信息加入httpPost中,接着同Get利用execute执行POST请求,然后调用getEntity()获得服务器返回的实体信息。


    part.3 本例中C/S双方工作机制分析

    如下图:本文的安卓客户端分别用上述讲的四种方法向本地的Java Web进行访问,图中显示为执行URI_GET请求时的客户端和服务器后台的效果:点击URI_GET按钮-->客户端启动Quest(1)线程使用Func1()进行Get请求-->当请求结束通过Message将从Func1()返回的服务器返回的Response字符串传送给消息接收句柄,在消息接受句柄中进行对UI中的TextView更新,显示返回结果。而服务器端如part1中介绍,当接收到客户端的POST或GET请求时,都会委托给execute函数来处理,并用PrintWriter out = response.getWriter();将消息发给客户端。


    part.4 拓展知识

    拓展知识均来自网络:请支持原创作者。
    http://www.apkbus.com/android-13575-1-1.html
    第一种版本:

    • HTTP 定义了与服务器交互的不同方法,最基本的方法是 GET 和 POST。
    • 事实上 GET 适用于多数请求,而保留 POST 仅用于更新站点。根据 HTTP 规范,GET 用于信息获取,而且应该是 安全的和 幂等的。所谓安全的意味着该操作用于获取信息而非修改信息。换句话说,GET 请求一般不应产生副作用。幂等的意味着对同一 URL 的多个请求应该返回同样的结果。完整的定义并不像看起来那样严格。从根本上讲,其目标是当用户打开一个链接时,它可以确信从自身的角度来看没有改变资源。 比如,新闻站点的头版不断更新。虽然第二次请求会返回不同的一批新闻,该操作仍然被认为是安全的和幂等的,因为它总是返回当前的新闻。反之亦然。
    • POST 请求就不那么轻松了。POST 表示可能改变服务器上的资源的请求。仍然以新闻站点为例,读者对文章的注解应该通过 POST 请求实现,因为在注解提交之后站点已经不同了(比方说文章下面出现一条注解);
    • 在FORM提交的时候,如果不指定Method,则默认为GET请求,Form中提交的数据将会附加在url之后,以?分开与url分开。字母数字字符原 样发送,但空格转换为“+“号,其它符号转换为%XX,其中XX为该符号以16进制表示的ASCII(或ISO Latin-1)值。GET请求请提交的数据放置在HTTP请求协议头中,而POST提交的数据则放在实体数据中;
    • GET方式提交的数据最多只能有1024字节,而POST则没有此限制。 

    第二种版本:

    • get是从服务器上获取数据,post是向服务器传送数据。
    • 在客户端,Get方式在通过URL提交数据,数据在URL中可以看到;POST方式,数据放置在HTML HEADER内提交。
    • 对于get方式,服务器端用Request.QueryString获取变量的值,对于post方式,服务器端用Request.Form获取提交的数据。
    • GET方式提交的数据最多只能有1024字节,而POST则没有此限制。
    • 安全性问题。正如在(1)中提到,使用 Get 的时候,参数会显示在地址栏上,而 Post 不会。所以,如果这些数据是中文数据而且是非敏感数据,那么使用 get;如果用户输入的数据不是中文字符而且包含敏感数据,那么还是使用 post为好。

    第三种版本:

    • Get是用来从服务器上获得数据,而Post是用来向服务器上传递数据。
    • Get将表单中数据的按照variable=value的形式,添加到action所指向的URL后面,并且两者使用“?”连接,而各个变量之间使用 “&”连接;Post是将表单中的数据放在form的数据体中,按照变量和值相对应的方式,传递到action所指向URL。
    • Get是不安全的,因为在传输过程,数据被放在请求的URL中,而如今现有的很多服务器、代理服务器或者用户代理都会将请求URL记录到日志文件中,然后 放在某个地方,这样就可能会有一些隐私的信息被第三方看到。另外,用户也可以在浏览器上直接看到提交的数据,一些系统内部消息将会一同显示在用户面前。 Post的所有操作对用户来说都是不可见的。
    • Get传输的数据量小,这主要是因为受URL长度限制;而Post可以传输大量的数据,所以在上传文件只能使用Post(当然还有一个原因,将在后面的提到)。
    • Get限制Form表单的数据集的值必须为ASCII字符;而Post支持整个ISO10646字符集。
    • Get是Form的默认方法。

    相关链接

    此外推荐一些链接帮助更好理解安卓GET和POST请求:

    1、我的漫漫程序之旅:[http://www.blogjava.net/supercrsky/articles/247449.html]

    内容提示:给出了JDK中的URLConnection参数详解,写的很详细,能帮助理解URLConnection

    2、pandazxx的专栏:[http://blog.csdn.net/pandazxx/article/details/1657109]

    内容提示:Http学习之使用HttpURLConnection发送post和get请求 ,有例子,有注释

    3、上述工程C/S代码:[http://pan.baidu.com/s/1qWqNUos]

    4、上述工程GitHub:[https://github.com/beautifulzzzz/Android/tree/master/HTTP_POST_GET]

  • 相关阅读:
    [Daily Coding Problem 223] O(1) space in order traversal of a binary tree
    [Daily Coding Problem 224] Find smallest positive integer that is not the sum of a subset of a sorted array
    Sliding Window Algorithm Questions
    Sweep Line Algorithm Summary
    Palindrome Algorithm Questions
    Core Data Structures, Algorithms and Concepts
    [LeetCode] 1000. Minimum Cost to Merge Stones
    UVA 253 Cube painting(枚举 模拟)
    Codeforces 821C Okabe and Boxes
    Codeforce 741B Arpa's weak amphitheater and Mehrdad's valuable Hoses(并查集&分组背包)
  • 原文地址:https://www.cnblogs.com/zjutlitao/p/4353355.html
Copyright © 2020-2023  润新知