首页 > 代码库 > Android 使用Okhttp/Retrofit持久化cookie的简便方式

Android 使用Okhttp/Retrofit持久化cookie的简便方式

首先cookie是什么就不多说了,还是不知道的话推荐看看这篇文章 
Cookie/Session机制详解 
深入解析Cookie技术

为什么要持久化cookie也不多说了,你能看到这篇文章代表你有这个需求。

cookie简单来说就是服务器在客户端中保存的键值对,比如说早期的购物车,保持登陆状态都是使用的cookie。 
但是cookie的功能是依赖于浏览器的,大多数浏览器都有管理cookie的功能。当然,你也能通过设置禁止掉这项功能,毕竟cookie是很容易泄露用户隐私的

上面也说了cookie功能依赖于客户端,很明显,在开发app的时候我们也要手动管理cookie了。

 

持久化cookie之前,我们最好了解一下cookie是怎么传输的,看完之后我想你就能使用很简单的方式去保持cookie了。

1. cookie是怎么传输的

Cookie使用HTTPHeader传递数据。 
Cookie机制定义了两种报头:Set-Cookie报头和Cookie报头。

Set-Cookie报头包含于Web服务器的响应头(ResponseHeader)中 
Cookie报头包含在浏览器客户端请求头(ReguestHeader)中

Cookie的运行过程如图所示,具体分析如下 
技术分享

2. 简单粗暴持久化cookie的方式

方法来源于这http://tsuharesu.com/handling-cookies-with-okhttp/

看了cookie的传输原理之后我们就可以用一个简单的方式持久化cookie了,那就是通过okhttp的intercept方式手动往header中写入cookie或者获取cookie。

/** * This interceptor put all the Cookies in Preferences in the Request. * Your implementation on how to get the Preferences MAY VARY. * <p> * Created by tsuharesu on 4/1/15. */public class AddCookiesInterceptor implements Interceptor {    @Override    public Response intercept(Chain chain) throws IOException {        Request.Builder builder = chain.request().newBuilder();        HashSet<String> preferences = (HashSet) Preferences.getDefaultPreferences().getStringSet(Preferences.PREF_COOKIES, new HashSet<>());        for (String cookie : preferences) {            builder.addHeader("Cookie", cookie);            Log.v("OkHttp", "Adding Header: " + cookie); // This is done so I know which headers are being added; this interceptor is used after the normal logging of OkHttp        }        return chain.proceed(builder.build());    }}
/** * This Interceptor add all received Cookies to the app DefaultPreferences. * Your implementation on how to save the Cookies on the Preferences MAY VARY. * <p> * Created by tsuharesu on 4/1/15. */public class ReceivedCookiesInterceptor implements Interceptor {    @Override    public Response intercept(Chain chain) throws IOException {        Response originalResponse = chain.proceed(chain.request());        if (!originalResponse.headers("Set-Cookie").isEmpty()) {            HashSet<String> cookies = new HashSet<>();            for (String header : originalResponse.headers("Set-Cookie")) {              cookies.add(header);            }            Preferences.getDefaultPreferences().edit()                    .putStringSet(Preferences.PREF_COOKIES, cookies)                    .apply();        }        return originalResponse;    }}
/** * Somewhere you create a new OkHttpClient and use it on all your requests. */OkHttpClient okHttpClient = new OkHttpClient();okHttpClient.interceptors().add(new AddCookiesInterceptor());okHttpClient.interceptors().add(new ReceivedCookiesInterceptor());

简单的cookie持久化可以用这种方式,创建一个单例的client,全局都使用这个client请求接口。当然你也可以每次new一个client重新设置intercept……

还要记住的是,cookie是针对于域名存储的。比如:www.baidu.com和image.baidu.com存储的cookies都是不一样的…… 
如果你的app真的需要同时访问两个域名的接口,并且两个都需要持久化cookie,那么记得做判断(比如用域名作为key存储cookie到sharepreference中)。否则两个域名的cookie会互相覆盖掉……

3. 本体在这里,okhttp 3.0 cookie的管理

相关原理可以看看这篇文章 OkHttp3之Cookies管理及持久化

然后,持久化管理cookie可以直接使用这个开源库,简单到爆炸…… 
https://github.com/franmontiel/PersistentCookieJar

当然,你也可以使用鸿洋大神的okhttputils,该库也提供了cookie的持久化管理工具。用起来和上面一样方便

Android 使用Okhttp/Retrofit持久化cookie的简便方式