OkHttp学习系列二:谈谈Android中使用的坑
近期,对一个Eclipse下开发的Android应用进行重构,原先使用的网络框架是AsyncHttpClient。众所周知,HttpClient已经在Android 6.0中已经被官方remove了。官方推荐咱们使用的 HttpURLConnection
。当然如果你执意或者不得已要用它,可以通过在build.gradle中加入下面的代码获取。
android { useLibrary 'org.apache.http.legacy'}
话题跑偏了。。。我是来说坑的。
回调方法不会自动到主线程中运行
就如上一篇OkHttp学习系列一:入门和简单使用所说的,OkHttp的设计是适合Java和Android程序的,但不是专门为Android应用设计。所以关于Android中的两个原则,它是不关心的。哪两个呢?
- 不能在主线程(UI主线程)发起网络;
- UI操作应该在UI线程中执行;
查看OkHttp3的源码发现:使用enqueue方法进行网络访问时,OkHttp会在线程池中调用异步任务进行网络访问和回调,所以原则1是满足的。但是下面的代码还是有问题,报了异常:CalledFromWrongThreadException
。为什么?
OkHttpClient client = new OkHttpClient();Request request = new Request.Builder().url("http://lshare.me").build();client.newCall(request).enqueue(new Callback() { @Override public void onResponse(Call call, Response response) throws IOException { Toast.makeText(TroubleDetailActivity.this,"Success",Toast.LENGTH_SHORT).show(); } @Override public void onFailure(Call call, IOException e) { }});
因为Toast涉及UI,必须在UI线程中执行,而OkHttp不会自动到主线程中执行回调方法的,违反原则2。so,下面才是正确姿势。
OkHttpClient client = new OkHttpClient();Request request = new Request.Builder().url("http://lshare.me").build();client.newCall(request).enqueue(new Callback() { @Override public void onResponse(Call call, Response response) throws IOException { runOnUiThread(new Runnable() {//这是Activity的方法,会在主线程执行任务 @Override public void run() { Toast.makeText(TroubleDetailActivity.this, "Success", Toast.LENGTH_SHORT).show(); } }); } @Override public void onFailure(Call call, IOException e) { }});
没有帮我缓存并发送Cookie给服务器
使用上面的代码访问网络,会发现本来已经在登录页面登录成功了,然而在向服务器请求数据时,服务器告诉我:你这家伙还没登录啊。我醉了,明明登录成功了。仔细一想,会不会Cookie没有带过去,服务器可是只认Cookie不认人的,虽然我还是有那么点帅。用Fiddle
抓包一看,吓死本宝宝了,果然没有发送过去。于是,赶紧Google了一下,看到网上说的:
- OkHttp 3.0开始,默认不保存Cookie,要自己使用
CookieJar
来保存Cookie。我用的就是3.0,命中。 - 使用Builder来构建OkHttpClient才能设置
CookieJar
。我是直接new的,命中; - 使用下面的代码可以帮你自动管理Cookie。下面的代码摘自OkHttp3之Cookies管理及持久化
OkHttpClient client = new OkHttpClient.Builder() .cookieJar(new CookieJar() { private final HashMap> cookieStore = new HashMap<>(); @Override public void saveFromResponse(HttpUrl url, List cookies) { cookieStore.put(url.host(), cookies); } @Override public List loadForRequest(HttpUrl url) { List cookies = cookieStore.get(url.host()); return cookies != null ? cookies : new ArrayList(); } }) .build();
第3个说法就有问题了,首先这个cookieStore
的key究竟是HttpUrl
呢,还是String。因为代码中,创建时使用
HttpUrl,而get和put时却是传入
url.host()`,而这是个String,不科学!博主走心了。
尝试下,使用HttpUrl作为key,修改put和get,传入url,用fiddler抓包,发现问题依旧。又用了String作为key,修改cookieStore
的定义,改key为String,用fiddler再次试了下。好激动,居然成啦!
//记录下正确姿势OkHttpClient mHttpClient = new OkHttpClient.Builder().cookieJar(new CookieJar() { private final HashMap> cookieStore = new HashMap<>(); //Tip:這裡key必須是String @Override public void saveFromResponse(HttpUrl url, List cookies) { cookieStore.put(url.host(), cookies); } @Override public List loadForRequest(HttpUrl url) { List cookies = cookieStore.get(url.host()); return cookies != null ? cookies : new ArrayList(); }}).build();
为什么?
为什么不能用HttpUrl?看起来代码似乎没有什么不妥啊
看下HttpUrl的源代码,官方给我这个案例。说明了一个HttpUrl由scheme、host、pathSegment、queryParameter四部分组成。
HttpUrl url = new HttpUrl.Builder() .scheme("https") .host("www.google.com") .addPathSegment("search") .addQueryParameter("q", "polar bears") .build();
这不很明显吗?我们在登录页面发送的url和获取主页面数据时请求的url是不同的。我们使用前一次的url保存cookie在hashMap中,在请求主页面数据时OkHttp用新的url从hashMap取不到cookie,所以服务器不认你了,以为你没有登录。而url.host()
是每次都一样的,这是重点!
更新日志:
- 3月26日:感谢@_醉生梦死 对本文的修正。
更多相关文章
- 4 行代码实现 ANDROID 快速文件下载
- ART模式下dex2oat出错导致系统无法正常启动
- AOP在Android中最佳用法
- JPush+SAE+J2EE实现微信公众平台账号服务
- 《Android(安卓)第一行代码》十一章 Service学习笔记
- EventBus3.x的正确打开方式
- iOS开发:几种静态扫描工具的使用与对比
- ffplay2 android 版正式公布
- android中如何添加一个监听按钮,点击之后从一个activity跳转到另