这里有新鲜出炉的精品教程,程序狗速度看过来!
Android 是一种基于 Linux 的自由及开放源代码的操作系统,主要使用于移动设备,如智能手机和平板电脑,由 Google 公司和开放手机联盟领导及开发。尚未有统一中文名称,中国大陆地区较多人使用 "安卓" 或 "安致"。
本篇文章主要介绍了详解 Android 中 OkHttp3 的例子和在子线程更新 UI 线程的方法 ,非常具有实用价值,需要的朋友可以参考下
okHttp 用于 android 的 http 请求。据说很厉害,我们来一起尝尝鲜。但是使用 okHttp 也会有一些小坑,后面会讲到如何掉进坑里并爬出来。
首先需要了解一点,这里说的 UI 线程和主线程是一回事儿。就是唯一可以更新 UI 的线程。这个只是点会在给 okHttp 填坑的时候用到。而且,这个内容本身在日常的开发中也经常用到,值得好好学一学。
okHttp 发起同步请求
第一个列子是一个同步请求的例子。
- private void performSyncHttpRequest() {
- OkHttpClient client = new OkHttpClient();
- Request request = new Request.Builder()
- .url("http://www.baidu.com")
- .build();
- Call call = client.newCall(request);
- Response response = call.execute();
- }
但是这样的直接在 android 的主线程里调用一个网络请求的方法是行不通的,直接抛出 UI Thread 请求网络的异常。所以我们这里为了可以掩饰要做一点小小的改动。把请求写成同步请求的方式,但是放在一个 worker 线程里异步的做这个操作。
- private Handler requestHandler = new Handler() {
- @Override
- public void handleMessage(Message msg) {
- switch (msg.what) {
- case REQUEST_SUCCESS:
- Toast.makeText(MainActivity.this, "SUCCESSFUL", Toast.LENGTH_SHORT).show();
- break;
- case REQUEST_FAIL:
- Toast.makeText(MainActivity.this, "request failed", Toast.LENGTH_SHORT).show();
- break;
- default:
- super.handleMessage(msg);
- }
- }
- };
- private void performSyncHttpRequest() {
- Runnable requestTask = new Runnable() {
- @Override
- public void run() {
- Message msg = requestHandler.obtainMessage();
- try {
- OkHttpClient client = new OkHttpClient();
- Request request = new Request.Builder()
- .url("http://www.baidu.com")
- .build();
- Call call = client.newCall(request);
- // 1
- Response response = call.execute();
- if (!response.isSuccessful()) {
- msg.what = REQUEST_FAIL;
- } else {
- msg.what = REQUEST_SUCCESS;
- }
- } catch (IOException ex) {
- msg.what = REQUEST_FAIL;
- } finally {
- // send the message
- // 2
- msg.sendToTarget();
- }
- }
- };
- Thread requestThread = new Thread(requestTask);
- requestThread.start();
- }
所以同步的请求都是这么做的 Response response = call.execute();。
1. 发起同步请求之前先新初始化一个 OkHttpClient。然后是具体的请求,用请求 builder 来创建这个 Request。我们这里为了简单 url 就是 * http://www.baidu.com * 了。接下来用前面初始化好的 client 发起一个 call:Call call = client.newCall(request);。最后执行这个 call:Response response = call.execute(); 并获得请求的 response。
2. 这一部分的可以暂时不要关注。因为这个例子只是为了能以运行起来的方式展示 okHttp 如何发起同步请求。
okHttp 发起异步请求
既然 android 本身不支持发起同步请求,当然也没人要发起同步请求。这么做是能导致严重的用户体验问题。想象一下,如果你有一个瀑布流,然后瀑布流里全部显示的都是图片。现在用户要不断地往下翻看瀑布流的图片。如果这些图片都用同步请求的话,什么时候可以翻一页不说,系统的 ANR 早就跳出来了。
所以我们就探究一下如何发起一个异步的请求。
- private void performAsyncHttpRequest() {
- OkHttpClient client = new OkHttpClient();
- Request request = new Request.Builder()
- .url("http://www.baidu.com")
- .build();
- Call call = client.newCall(request);
- // 1
- call.enqueue(new Callback() {
- // 2
- @Override
- public void onFailure(Call call, IOException e) {
- //Toast.makeText(MainActivity.this, e.getMessage(), Toast.LENGTH_SHORT).show();
- if (Looper.getMainLooper().getThread() == Thread.currentThread()) {
- Log.d(TAG, "Main Thread");
- } else {
- Log.d(TAG, "Not Main Thread");
- }
- }
- @Override
- public void onResponse(Call call, final Response response) throws IOException {
- // 3
- if (Looper.getMainLooper().getThread() == Thread.currentThread()) {
- Log.d(TAG, "Main Thread");
- } else {
- Log.d(TAG, "Not Main Thread");
- }
- }
- });
- }
1. 同步请求用 execute 方法,异步就用 call.enqueue(new Callback() 方法。
2. 这个 Callback 接口提供了两个方法,一个是 onFailure,一个是 onResponse。这两个方法分别在请求失败和成功的时候调用。
3. 本来一切都似乎应该很简单。网络请求成功或者失败直接在界面更新了。但是木有想到这样会抛异常。然后看了看发现原来 onFailure 和 onResponse 两个方法不是在主线程执行。打印出来的 log 是:okhttp.demo.com.okhttpdemo D/###okHttp: Not Main Thread。
所以要在主线程中更新 view 只好想别的办法了。在 worker 线程里更新主线程会抛异常。一般来说有这么几个方法在子线程里更新 view。
在子线程更新 UI 线程
一、Activity 的 runOnUiThread 方法
在 Activity 中有这么一个方法 runOnUiThread。这个方法需要一个 Runnable 实例作为参数。
- MainActivity.this.runOnUiThread(new Runnable() {
- @Override
- public void run() {
- Log.d(TAG, "code: ");
- Toast.makeText(MainActivity.this, String.valueOf(response.code()), Toast.LENGTH_SHORT).show();
- }
- });
二、View 的 post 方法
View 的 post 方法也是一样,扔一个 Runnable 的实例进去。然后就在主线程执行了。Toast 肯定是没有这个方法的。
- MainActivity.this.mView.post(new Runnable() {
- public void run() {
- Log.d("UI thread", "I am the UI thread");
- }
- });
三、其他
1. 用 Handler,这个前面的 okHttp 同步请求的例子可以用。
2.AsyncTask, 有两个方法可以在主线程中执行:onProgressUpdate 和 onPostExecute。这里我们并不是要更新进度,所以考虑的是后一个方法。
- private class BackgroundTask extends AsyncTask < String,
- Void,
- Bitmap > {
- protected void onPostExecute(Bitmap result) {
- Log.d("UI thread", "I am the UI thread");
- }
- }
综合以上,更新 UI 线程的方法里最后说到的 Handler 方法和 AsyncTask 都太重。尤其是 AsyncTask。还要继承实现一堆的方法之后才可以能达到目的,同时还和我们要用的 okHttp 的使用方法很多不兼容的地方。
所以我们只考虑前面的两种。但是两种方法其实是不一样的。当然,这里并不是说方法的名字不一样。我们来看看 android 的源代码,这两个方法是如何实现的。
- public final void runOnUiThread(Runnable action) {
- if (Thread.currentThread() != mUiThread) {
- mHandler.post(action);
- } else {
- action.run();
- }
- }
- // mHandler.post(action); 之post方法的实现
- public final boolean post(Runnable r) {
- return sendMessageDelayed(getPostMessage(r), 0);
- }
方法 runOnUiThread 最后会调用 Handler 的 sendMessageDelayed。但是这里只 delay 了 0。也就是方法传到这里的时候会立即执行 runOnUiThread 的参数 Runnable 实例会立即执行。
下面看看 View 的 post 方法:
- public boolean post(Runnable action) {
- final AttachInfo attachInfo = mAttachInfo;
- if (attachInfo != null) {
- return attachInfo.mHandler.post(action);
- }
- // Assume that post will succeed later
- ViewRootImpl.getRunQueue().post(action);
- return true;
- }
一般会执行的是 ViewRootImpl.getRunQueue().post(action);。也就是 Runnable 的实例只是添加到了事件队列中,按照顺序执行。并不一定会立即执行。
我们探讨了那么多,最后就使用 runOnUiThread 来更新界面,也就是方法一了。
- call.enqueue(new Callback() {
- @Override
- public void onFailure(Call call, IOException e) {
- final String errorMMessage = e.getMessage();
- if (Looper.getMainLooper().getThread() == Thread.currentThread()) {
- Log.d(TAG, "Main Thread");
- } else {
- Log.d(TAG, "Not Main Thread");
- }
- MainActivity.this.runOnUiThread(new Runnable() {
- @Override
- public void run() {
- Toast.makeText(MainActivity.this, errorMMessage, Toast.LENGTH_SHORT).show();
- }
- });
- }
- @Override
- public void onResponse(Call call, final Response response) throws IOException {
- if (Looper.getMainLooper().getThread() == Thread.currentThread()) {
- Log.d(TAG, "Main Thread");
- } else {
- Log.d(TAG, "Not Main Thread");
- }
- MainActivity.this.runOnUiThread(new Runnable() {
- @Override
- public void run() {
- Log.d(TAG, "code: ");
- Toast.makeText(MainActivity.this, String.valueOf(response.code()), Toast.LENGTH_SHORT).show();
- }
- });
- }
- });
无论请求成功还是失败,都弹出一个 Toast。用 MainActivity.this.runOnUiThread 在 UI 线程中弹出这个 Toast。
来源: http://www.phperz.com/article/17/0528/335583.html