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。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
本文向大家介绍解决PySide+Python子线程更新UI线程的问题,包括了解决PySide+Python子线程更新UI线程的问题的使用技巧和注意事项,需要的朋友参考一下 在我开发的系统,需要子线程去运行,然后把运行的结果发给UI线程,让UI线程知道运行的进度。 首先创建线程很简单 之后我发现用子线程去调用UI线程是行不通的,只能通过信号和槽来实现,于是 首先,定义一个类,让他实现PySide.Q
本文向大家介绍Android主线程和子线程区别详解,包括了Android主线程和子线程区别详解的使用技巧和注意事项,需要的朋友参考一下 主线程和子线程的区别 每个线程都有一个唯一标示符,来区分线程中的主次关系的说法。 线程唯一标示符:Thread.CurrentThread.ManagedThreadID; UI界面和Main函数均为主线程。 被Thread包含的“方法体”或者“委托”均为子线程。
问题内容: 我刚刚开始进行android开发,并且更新UI确实使我很烦:/ 到目前为止,这是我一直在努力的工作- 它工作正常,但是每个人都说要在主线程中做图形,所以我正在尝试(但失败)将其传递给另一个线程。麻烦的是,我真的不知道怎么做,因为我从未真正使用过线程。 Google给出的有关使用Threads的示例似乎并不十分清楚,我也无法真正按照自己的意愿去做。我可以请某人在这里给我一个最基本的例子,
本文向大家介绍Android子线程与更新UI问题的深入讲解,包括了Android子线程与更新UI问题的深入讲解的使用技巧和注意事项,需要的朋友参考一下 前言 在Android项目中经常有碰到这样的问题,在子线程中完成耗时操作之后要更新UI,下面就自己经历的一些项目总结一下更新的方法。话不多说了,来一起看看详细的介绍吧 引子: 情形1 运行结果:正常运行!!! 情形二 运行结果:异常 and
问题内容: 据我所知,在android中,当我们运行进程时,它们始于Main线程。当我们做一些繁重的工作时,我们使用了一个新线程。如果我们想修改UI外观,可以使用在UI上运行。 有人可以向我解释这些线程的作用以及如何使用它们吗? 问题答案: UI线程和主线程只是同一线程的不同名称。 应用程序的所有UI膨胀都在此主线程上完成。之所以将“较重”的工作委派给其他线程,是因为我们不希望这些操作减慢UI的响
我正在编写一个Android应用程序,它连接到蓝牙设备,读取设备发送的数据,将其添加到AChartEngine图形中,并在文本视图中显示数据。 我的Bluetooth代码与BluetoothChat示例代码中的线程实现非常相似(它与SDK一起提供)。我可以在LogCat中看到< code>ConnectedThread循环正在执行并因此获得新数据,但我的TextView在7行之后停止更新,图形间歇