Volley 添加Cookie和获取服务器返回的一条或多条Cookie

来源:互联网 发布:亲戚借身份证开淘宝店 编辑:程序博客网 时间:2024/05/16 15:17

之前项目里请求网络的代码都是采用原生的语句,如HttpURLConnection或者HttpClient,这种方式请求灵活度高,但是访问网络的速度慢,特别是在网络不好的情况下情况更糟,

而Volley是google官方极力推崇的一款网络请求框架,具体好处自不用多说.且因为它的继承性,在灵活度上也比原生的请求代码好一些,我们可以继承父类Request<T>来实现我们自己的xxxRequest.

当采用post方式请求数据时,我们一般都要添加请求参数,和一些header信息,cookie信息自然也不例外, 而cookie的作用主要就是保存用户的一些身份信息等,如一个软件在使用的过程.从登陆账号成功的那一刻起,到关闭软件退出的那一刻结束,这段时间内在软件的任何有关网络的post请求操作都要带上cookie信息而cookie信息是服务器端返回的,所以软件在第一次登陆的时候肯定是没有cookie的,当登陆成功后,服务器端返回一段保存在"Set-Cookie"头字段里的cookie信息,客户端拿到这段数据以后,一般都要对返回的Cookie进行抽取,把真正的cookie信息抽取出来然后保存在sharedPreference里或者数据库里,然后在软件的使用过程中,所有需要cookie的地方都从sharedPreference里或者数据库里取,但是cookie在服务端的存储也是有时间限制的,所以以后的每次登陆都要在登陆成功后把最新的cookie信息保存一遍,这样能保证每次用的cookie都是有效的.

Volley框架中有获取cookie的方式,只要重写parseNetWorkResponse,该回调方法返回一个NetWorkResponse对象,而NetWorkResponse.headers方法就可以获取到所有的响应头字段,所有的头字段封装在Map集合里,如下代码:

[java] view plain copy
  1. Map<String, String> headers = response.headers;//获取所有头字段  
  2. String cookies = headers.get("Set-Cookie");//获取Cookie头字段  
  3. String data = new String(response.data, "UTF-8");//获取服务器返回的数据  

大家也看到了,通过header.get("Set-Cookie")方法获取服务器端返回的cookie信息,但是有个不足,Volley默认只返回第一条Cookie信息,如果服务端返回了多条cookie信息,而我们只能获取一条,这显然不行,所以我们需要修改源码,之前也在网络上找了一些资料,但是要么就是说继承Request自己实现的,要么就是自己修改源码但是都没说明白的,但是继承Request自己实现这种方式还是解决不了获取所有cookie信息的问题,因为只要是继承request,都需要重写两个方法,一个deliverResponse(String response),方法,该方法一般只回调mListener.onResponse(response)即可,另一个必须要重写的就是parseNetworkResponse(NetworkResponse response)方法,该方法就是解析NetworkResponse ,并把解析好的结果返回给deliverResponse(String response),而解析好的结果一般也就是我们需要展示到界面上的东西,但是我们解析的NetworkResponse 是Volley在添加完头字段后返回的,也就是说,Volley已经把cookie信息的第一条数据取出来后,才返回的NetworkResponse ,所以即使继承request自己实现,还是只会获取到1条cookie信息,并不能获取所有的cookie,如下Volley源码,具体位置在com.Android.volley.toolbox.HurlStack类中的performRequest方法中可以看到

[java] view plain copy
  1.  @Override    
  2.    public HttpResponse performRequest(Request<?> request, Map<String, String> additionalHeaders)    
  3.            throws IOException, AuthFailureError {    
  4.        String url = request.getUrl();    
  5.        HashMap<String, String> map = new HashMap<String, String>();    
  6.        if (mUserAgent != null) {    
  7.            map.put("User-Agent", mUserAgent);    
  8.        }    
  9.        map.putAll(request.getHeaders());    
  10.        map.putAll(additionalHeaders);    
  11.        if (mUrlRewriter != null) {    
  12.            String rewritten = mUrlRewriter.rewriteUrl(url);    
  13.            if (rewritten == null) {    
  14.                throw new IOException("URL blocked by rewriter: " + url);    
  15.            }    
  16.            url = rewritten;    
  17.        }    
  18.        URL parsedUrl = new URL(url);    
  19.        HttpURLConnection connection = openConnection(parsedUrl, request);    
  20.        for (String headerName : map.keySet()) {    
  21.            connection.addRequestProperty(headerName, map.get(headerName));    
  22.        }    
  23.        setConnectionParametersForRequest(connection, request);    
  24.        // Initialize HttpResponse with data from the HttpURLConnection.    
  25.        ProtocolVersion protocolVersion = new ProtocolVersion("HTTP"11);    
  26.        int responseCode = connection.getResponseCode();    
  27.        if (responseCode == -1) {    
  28.            // -1 is returned by getResponseCode() if the response code could not be retrieved.    
  29.            // Signal to the caller that something was wrong with the connection.    
  30.            throw new IOException("Could not retrieve response code from HttpUrlConnection.");    
  31.        }    
  32.        StatusLine responseStatus = new BasicStatusLine(protocolVersion,    
  33.                connection.getResponseCode(), connection.getResponseMessage());    
  34.        BasicHttpResponse response = new BasicHttpResponse(responseStatus);    
  35.        response.setEntity(entityFromConnection(connection));    
  36.        //////////////////////////////////////////////////////////////////    
  37.     (Entry<String, List<String>> header : connection.getHeaderFields().entrySet()) {    
  38.     if (header.getKey() != null) {    
  39.         ////////////看这里~~/////////////////////////                
  40.             Header h = new BasicHeader(header.getKey(), header.getValue().get(0));    
  41.             response.addHeader(h);    
  42.         }    
  43.     }    
  44.     return response;    
  45. }    


可以看到,虽然volley使用了for循环,但是却只获取第一条value,这就导致了如果cookie的value有多条数据,我们只能获取第一条,从这个原因入手,我们有个根本的解决方案,就是在获取所有头字段和它的值的时候,我们直接遍历每个键的所有值,然后把值依次添加到StringBuilder里,然后放到 Header h = new BasicHeader(header.getKey(),sb.toString());  

里,这样在调用parseNetworkResponse(NetworkResponse response)方法的时候,NetworkResponse里就存储了所有的cookie信息,我们直接获取即可.具体的代码如下

首先是修改获取value的地方:

[java] view plain copy
  1. @Override    
  2.    public HttpResponse performRequest(Request<?> request, Map<String, String> additionalHeaders)    
  3.            throws IOException, AuthFailureError {    
  4.        String url = request.getUrl();    
  5.        HashMap<String, String> map = new HashMap<String, String>();    
  6.        if (mUserAgent != null) {    
  7.            map.put("User-Agent", mUserAgent);    
  8.        }    
  9.        map.putAll(request.getHeaders());    
  10.        map.putAll(additionalHeaders);    
  11.        if (mUrlRewriter != null) {    
  12.            String rewritten = mUrlRewriter.rewriteUrl(url);    
  13.            if (rewritten == null) {    
  14.                throw new IOException("URL blocked by rewriter: " + url);    
  15.            }    
  16.            url = rewritten;    
  17.        }    
  18.        URL parsedUrl = new URL(url);    
  19.        HttpURLConnection connection = openConnection(parsedUrl, request);    
  20.        for (String headerName : map.keySet()) {    
  21.            connection.addRequestProperty(headerName, map.get(headerName));    
  22.        }    
  23.        setConnectionParametersForRequest(connection, request);    
  24.        // Initialize HttpResponse with data from the HttpURLConnection.    
  25.        ProtocolVersion protocolVersion = new ProtocolVersion("HTTP"11);    
  26.        int responseCode = connection.getResponseCode();    
  27.        if (responseCode == -1) {    
  28.            // -1 is returned by getResponseCode() if the response code could not be retrieved.    
  29.            // Signal to the caller that something was wrong with the connection.    
  30.            throw new IOException("Could not retrieve response code from HttpUrlConnection.");    
  31.        }    
  32.        StatusLine responseStatus = new BasicStatusLine(protocolVersion,    
  33.                connection.getResponseCode(), connection.getResponseMessage());    
  34.        BasicHttpResponse response = new BasicHttpResponse(responseStatus);    
  35.        response.setEntity(entityFromConnection(connection));    
  36.        for (Entry<String, List<String>> header : connection.getHeaderFields().entrySet()) {    
  37.           ///////////改成如下代码////////////////////////    
  38.        if (header.getKey() != null) {    
  39.                  StringBuilder builder = new StringBuilder();     
  40.          //获取一个key中的所有value     
  41.                  for(int i=0;i<header.getValue().size();i++){      
  42.                      sb.append(header.getValue().get(i));      
  43.                  }      
  44.                  Header h = new BasicHeader(header.getKey(),builder.toString());    
  45.          //原始代码就不要了    
  46.               //  Header h = new BasicHeader(header.getKey(), header.getValue().get(0));    
  47.                 response.addHeader(h);    
  48.             }    
  49.         }    
  50.         return response;    
  51.     }    


修改完毕以后,当我们使用StringRequest等其他request时,只要重写parseNetworkResponse方法,然后获取cookie信息即可,具体代码如下:

[java] view plain copy
  1. @Override    
  2.             protected Response<String> parseNetworkResponse(    
  3.                     NetworkResponse response) {    
  4.                 try {    
  5.                     Map<String, String> headers = response.headers;    
  6.                     String cookies = headers.get("Set-Cookie");    
  7.                     String data = new String(response.data, "UTF-8");    
  8.                     // 解析服务器返回的cookie值    
  9.                         String cookie = parseVolleyCookie(cookies);    
  10.             // 存储cookie  
  11.             SharedPreferenceUtil.putString(LoginActivity.this,"cookie", cookie);  
  12.             // 下面这句是把返回的数据传到onResponse回调里,一定要用这种方式写  
  13.             return Response.success(data,HttpHeaderParser.parseCacheHeaders(response));  
  14.         }   
  15.         catch (UnsupportedEncodingException e) {  
  16.             return Response.error(new ParseError(e));  
  17.             }  
  18.         }  

拿到cookie之后,我们看看返回的cookie是什么样的,如下:

[plain] view plain copy
  1. cookieEmployeeId=40288186519ea85901519ead34c50001; Path=/cookieSysUserId=40288186519ea85901519ead34ae0000; Path=/JSESSIONID=B6200ABA230422C2A0D434A2645FB400; Path=/wash-web/; HttpOnly  

这显然不是我们想要的cookie格式啊,标准的格式应该是这样的

[plain] view plain copy
  1. cookieSysUserId=40288186519ea85901519ead34ae0000;JSESSIONID=B6200ABA230422C2A0D434A2645FB400;cookieEmployeeId=40288186519ea85901519ead34c50001;  

没有Path=/,也没有什么/wash-web/; HttpOnly之类的东西才对,,所以我们下面需要做的就是解析这个cookie,把我们想要的数据抽取出来组合成我们理想的格式,可能有人问为什么会有path/之类的东西,产生这种情况的主要原因是因为服务器端程序在被访问的时候,会根据你的身份开放相应的功能,比如A功能是放在a路径下,产生的cookie是a路径下的,B功能放在b路径下,产生的cookie是b路径下的,所以返回的总cookie就有path/等问题,产生的原因说明完毕,我们就开始解析,解析也是具有针对性的,应该有通用的解析方法,不过目前还没想好,如下代码:

[java] view plain copy
  1. /* 
  2.  * 方法的作用: 解析volley返回cookie   
  3.  */    
  4. public String parseVooleyCookie(String cookie) {    
  5.     StringBuilder sb = new StringBuilder();    
  6.     String[] cookievalues = cookie.split(";");    
  7.     for (int j = 0; j < cookievalues.length; j++) {    
  8.         String[] keyPair = cookievalues[j].split("/");    
  9.         for (int i = 0; i < keyPair.length; i++) {    
  10.             if (keyPair.length == 2) {    
  11.                 if (keyPair[1].contains("cookieSysUserId")    
  12.                         || keyPair[1].contains("JSESSIONID")) {    
  13.                     sb.append(keyPair[1]);    
  14.                     sb.append(";");    
  15.                     break;    
  16.                 }    
  17.             } else {    
  18.                 if (keyPair[0].contains("cookieEmployeeId")) {    
  19.                     sb.append(keyPair[0]);    
  20.                     sb.append(";");    
  21.                     break;    
  22.                 }    
  23.             }    
  24.         }    
  25.     }    
  26.     return sb.toString();    
  27. }    

解析之后就和我们理想的cookie格式一样了:
[plain] view plain copy
  1. cookieSysUserId=40288186519ea85901519ead34ae0000;JSESSIONID=B6200ABA230422C2A0D434A2645FB400;cookieEmployeeId=40288186519ea85901519ead34c50001;  

OK,说完如何获取和保存服务器返回的多条cookie信息,接下来就是如何把这些cookie信息再提交给服务器审核,因为cookie信息是存放在头字段里的(响应头和请求头都有),所以我们把cookie再放回头字段跟着请求参数一起提交给服务器即可,到这估计有人说一开始要把cookie获取下来,现在又要提交上去,这么麻烦,干脆不要cookie多方便,如果有这样的疑问,建议去网上搜搜关于cookie的说明,这里就不多说了.话说回来,我们如何提交这些头信息给服务器呢.其实更简单,volley已经把该回调接口给我们写好了,我们直接用即可,如下代码:
[java] view plain copy
  1. @Override    
  2.   public Map<String, String> getHeaders() throws AuthFailureError {    
  3.       // 给服务器上传cookie值    
  4.       Map<String, String> map = new HashMap<String, String>();    
  5.       String cookie = SharedPreferenceUtil.getString(    
  6.               LoginActivity.this"cookie");    
  7.       map.put("cookie", cookie);    
  8.       return map;    
  9.           
  10.   }    

这样就完成了Volley添加Cookie信息到服务器和获取服务器返回的多条cookie信息并保存的所有操作.我们并没有继承request,只是修改的几行原始代码就达到了我们的要求.

下面贴出完整的以StringRequest为例的请求操作代码:

[java] view plain copy
  1. private void getLoginFromServer(final Map<String, String> params) {  
  2.         StringRequest request = new StringRequest(Method.POST,  
  3.                 NetAccessAddress.getLoginUrlAction(), new Listener<String>() {  
  4.                     @Override  
  5.                     public void onResponse(String response) {  
  6.                         if (dialog != null && dialog.isShowing())  
  7.                             dialog.dismiss();  
  8.                         parseResult(response);  
  9.                     }  
  10.                 }, new Response.ErrorListener() {  
  11.                     @Override  
  12.                     public void onErrorResponse(VolleyError error) {  
  13.                         if (dialog != null && dialog.isShowing())  
  14.                             dialog.dismiss();                         
  15.                         Toast.makeText(getApplicationContext(),  
  16.                                 "服务器连接异常!" , Toast.LENGTH_SHORT)  
  17.                                 .show();  
  18.                     }  
  19.                 }) {  
  20.             @Override  
  21.             protected Map<String, String> getParams() throws AuthFailureError {  
  22.                 // 添加post请求参数  
  23.                 return params;  
  24.             }  
  25.   
  26.             @Override  
  27.             public Map<String, String> getHeaders() throws AuthFailureError {  
  28.                 // 添加头部信息,给服务器上传cookie值  
  29.                 Map<String, String> map = new HashMap<String, String>();  
  30.                 String cookie = SharedPreferenceUtil.getString(  
  31.                         LoginActivity.this"cookie");  
  32.                 map.put("cookie", cookie);  
  33.                 return map;  
  34.                   
  35.             }  
  36.   
  37.             @Override  
  38.             protected Response<String> parseNetworkResponse(  
  39.                     NetworkResponse response) {  
  40.                 try {  
  41.                     Map<String, String> headers = response.headers;  
  42.                     String cookies = headers.get("Set-Cookie");  
  43.                     String data = new String(response.data, "UTF-8");  
  44.                     // 解析服务器返回的cookie值  
  45.                     String cookie = parseVooleyCookie(cookies);  
  46.                     // 存储cookie  
  47.                     SharedPreferenceUtil.putString(LoginActivity.this,  
  48.                             "cookie", cookie);  
  49.                     // 下面这句是把返回的数据传到onResponse回调里  
  50.                     return Response.success(data,  
  51.                             HttpHeaderParser.parseCacheHeaders(response));  
  52.                 } catch (UnsupportedEncodingException e) {  
  53.                     return Response.error(new ParseError(e));  
  54.                 }  
  55.             }  
  56.         };  
  57.         MyApplication.getQueue().add(request);//添加请求到队列里  
  58.     }  

再说一句,现在大多数人使用Volley大概用的是Volley的jar包,而不是library文件,所以修改Volley.jar的话,当然也可以解压出来然后修改再打成jar,但是这样有点麻烦,不过要是使用Android Studio的话就不会存在这个问题,AndroidStudio连Android源码都可以修改,别说一个小小的jar包了,不过如果是用eclipse开发的,估计根本改不了jar包,所以这里提供一个

Volley的library文件,和volley.jar都以一样的效果,只不过library文件修改起来不管是使用AndroidStudio还是eclipse都特别方便,并且关于获取多条Cookie的代码我已经在这个library里改好了,其他的都和官方的一模一样.

下载地址如下:http://download.csdn.net/detail/qiang_xi/9422609