Android端实现Cookie机制

作者: CPPAlien | 来源:发表于2016-02-19 14:54 被阅读6083次

简介

Session是服务端验证客户端身份的一种机制。而Cookie是客户端存储的一种身份凭证,由服务端在回应的消息头中通过Set-Cookie字段“种”在客户端。以后每次客户端在向服务端请求时都会在消息头中带上Cookie字段。服务端就会根据Cookie的来判断此次请求是从哪个用户发过来的,是否是一次有效请求等。有关Cookie的标准定义可以参考 RFC6265。以下我们稍做介绍。

举个例子

首次打开浏览器请求http://www.baidu.com ,我们会得到一个response消息头如下格式。

首次请求百度返回的消息头的一部分
可以看到里面包含了多条Set-Cookie,Cookie是key-value形式的。每条Cookie都有一个效信息字段,有些还含有expires、domain和path等字段。其中的domain和path字段,区分了不同的cookie可以被哪些网页链接获得,如未设置domain,则使用当前的访问链接替代。比如上面的BD_HOME=0,我们在下表[本地cookie信息]中,一样可以看到它的domain为www.baidu.com,这是因为我们访问的就是这个地址。其中的expires和max-age定义了该Cookie的有效期,失效的Cookie在下次请求时不会被加入到Cookie头中。

请求后我们查看浏览器的Cookie表,可以看到这些Cookie信息被“种”下了。


本地cookie信息

此时再发送一条请求,可以看到请求的消息头如下:


请求消息头Cookie信息

不同网站的请求,会因为请求链接的不同,只能从浏览器取得属于自己的cookie,根据上文提到的domain和path字段来区分。关于匹配的详细规则还是可以查看RFC6265。这边我们用domain的匹配来稍微说下,在标准的5.1.3条目是有关domain matching的。

A string domain-matches a given domain string if at least one of the
   following conditions hold:

   o  The domain string and the string are identical.  (Note that both
      the domain string and the string will have been canonicalized to
      lower case at this point.)

   o  All of the following conditions hold:

      *  The domain string is a suffix of the string.

      *  The last character of the string that is not included in the
         domain string is a %x2E (".") character.

      *  The string is a host name (i.e., not an IP address).

从中我们可以看出domain要满足两点才能算是匹配,第一点,完全相同;如果第一点不满足则需要,请求链接的后缀包含存储的cookie的domain,并且不包含部分的最后一个字符还得是".",并且还是是个域名,而非IP地址。domain匹配成功后,还要匹配path,path的匹配要比domain复杂一些,具体可以查看标准的5.1.4。path匹配可以做到一个网站下的页面可以分别存储不同的cookie,也可以共享上层父页面的cookie,比较灵活。

客户端实现

Android自身所带的HttpUrlConnection方法是默认不开启Cookie存储的。不过我们可以用java提供的几个类来在Android中实现:
可以先在所有请求之前声明

CookieHandler.setDefault(new CookieManager(null, CookiePolicy.ACCEPT_ALL));

开启此开关后,每次请求的Set-Cookie信息都会被CookieManager处理。CookieManager又会使用第一个参数传入的CookieStore来处理Cookie的存储问题,因为此处传入了null,系统会默认调用一个基于CookieStore实现的CookieStoreImpl类来处理Cookie的存储,这个类的只有基于内存的存储,当进程被杀死后,下次再进入应用,保存的Cookie信息就会丢失。所以我们需要基于CookieStore这个接口实现一个具有内存和本地双存储机制的Cookie存储类。

可以参考:Fran Montiel实现的PersistentCookieStore 类:
https://gist.github.com/franmontiel/ed12a2295566b7076161

当解决了Cookie的存储后,我们就需要考虑以后我们的每次请求需要在请求的消息头中加入Cookie字段。以上用CookieStore存储下来的Cookie信息都会被保存成HttpCookie形式的信息。我们可以上面的百度例子中看到Cookie的组成样式,所以我们可以提取CookieStore中的信息并组合。

StringBuilder cookieBuilder = new StringBuilder();
String divider = "";
for (HttpCookie cookie : getCookies()) {
    cookieBuilder.append(divider);
    divider = ";";
    cookieBuilder.append(cookie.getName());
    cookieBuilder.append("=");
    cookieBuilder.append(cookie.getValue());
}
cookieString = cookieBuilder.toString();

然后把这个cookieString在以后的请求中加入到头中,如果你用HttpUrlConnection,你就可以

setRequestProperty("Cookie",cookieString);  

如果你需要让应用中打开的WebView页面也能共享使用Cookie,则需要使用android.webkit.CookieManager类来设置,简单式例代码如下。注意,第一个参数要使用链接的host部分。这样让web端的不同页面也可以共享这些cookie。

for (HttpCookie cookie : getCookies()) {
    CookieManager.getInstance().setCookie(Uri.parse(url).getHost(), cookie.toString());
}

OK,大功告成。

以下是我在Github上开源的一个基于Volley实现的网络层框架,也包括Cookie机制的Http请求,欢迎大家fork:
https://github.com/CPPAlien/DaVinci

作者简介
彭涛(@彭涛me) 致力于让技术变得易懂且有趣
个人博客:http://pengtao.me, GitHub地址:https://github.com/CPPAlien

相关文章

  • Android端实现Cookie机制

    简介 Session是服务端验证客户端身份的一种机制。而Cookie是客户端存储的一种身份凭证,由服务端在回应的消...

  • coookie的性能影响和弊端

    由于Cookie的实现机制,一旦服务器端向客户端发送了设置Cookie的意图,除非Cookie过期,否则客户端每次...

  • [Android]Cookie研究

    学习自:android WebView的cookie机制【WebView的cookie机制 】轻松搞定WebVie...

  • cookie session token

    cookie、session与token 一、详述概念 1、Cookie机制 cookie机制是采用在客户端保持状...

  • Android 开发中 Cookie 持久化研究

    参考:android WebView的cookie机制Android中Cookie获取、保存以及同步OkHttp3...

  • 2018-12-28

    <1> cookie cookie机制 Cookie技术是客户端的解决方案,Cookie就是由服务器发给客户端的特...

  • HTTP的Cookie机制和Session机制

    Cookie机制 什么是Cookie Cookie技术是客户端的解决方案,Cookie就是由服务器发给客户端的特殊...

  • 详谈cookie机制和session机制

    一、cookie机制和session机制的区别 具体来说cookie机制采用的是在客户端保持状态的方案,而sess...

  • cookie和session的区别

    一、cookie机制和session机制的区别 具体来说cookie机制采用的是在客户端保持状态的方案,而sess...

  • Cookie for iOS

    cookie机制: Cookie是在客户端存储服务器状态的一种机制,Web服务器可以通过Set-Cookie或者S...

网友评论

  • Avalon1:你好。有个问题想请教下,我Android用webview打开页面,页面的URL一直都在变得(因为每次进去要加上动态的授权码)然后调用webview.loadurl。这种怎么处理cookie哇,用默认的webview发现切换帐号的时候不对。
  • a451f101e876:安卓源码中有个CookieStoreImpl.java类,非公有,曾经尝试copy一份出来用,发现不是很完善,有些cookie不会被记录,只能自己改一改实现了,不太理解httpurlconnection默认不开启cookie的原因
  • f1db2a97aeb2:种下的cookey跨进城能用么
    CPPAlien:@丶飘风远 代码里有一个,PersistentCookieStore.java,里面是用SharedPref来存储Cookie,把MODE_PRIVATE 改成 MODE_MULTI_PROCESS,然后用其他进程来访问里面存储的Cookie
    f1db2a97aeb2:@丶飘风远 能说的全面点吗 求指教
    CPPAlien:@丶飘风远 DaVinci版本不可以,因为Cookie存储是用Mode_Private的,你可以fork一下,修改成可以夸进程的版本

本文标题:Android端实现Cookie机制

本文链接:https://www.haomeiwen.com/subject/zmcvkttx.html