詳解Android中OkHttp3的例子和在子線程更新UI線程的方法
okHttp用于android的http請(qǐng)求。據(jù)說(shuō)很厲害,我們來(lái)一起嘗嘗鮮。但是使用okHttp也會(huì)有一些小坑,后面會(huì)講到如何掉進(jìn)坑里并爬出來(lái)。
首先需要了解一點(diǎn),這里說(shuō)的UI線程和主線程是一回事兒。就是唯一可以更新UI的線程。這個(gè)只是點(diǎn)會(huì)在給okHttp填坑的時(shí)候用到。而且,這個(gè)內(nèi)容本身在日常的開(kāi)發(fā)中也經(jīng)常用到,值得好好學(xué)一學(xué)。
okHttp發(fā)起同步請(qǐng)求
第一個(gè)列子是一個(gè)同步請(qǐng)求的例子。
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的主線程里調(diào)用一個(gè)網(wǎng)絡(luò)請(qǐng)求的方法是行不通的,直接拋出UI Thread 請(qǐng)求網(wǎng)絡(luò)的異常。所以我們這里為了可以掩飾要做一點(diǎn)小小的改動(dòng)。把請(qǐng)求寫成同步請(qǐng)求的方式,但是放在一個(gè)worker線程里異步的做這個(gè)操作。
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(); }
所以同步的請(qǐng)求都是這么做的Response response = call.execute();。
1.發(fā)起同步請(qǐng)求之前先新初始化一個(gè)OkHttpClient。然后是具體的請(qǐng)求,用請(qǐng)求builder來(lái)創(chuàng)建這個(gè)Request。我們這里為了簡(jiǎn)單url就是*http://www.baidu.com*了。接下來(lái)用前面初始化好的client發(fā)起一個(gè)call:Call call = client.newCall(request);。最后執(zhí)行這個(gè)call:Response response = call.execute();并獲得請(qǐng)求的response。
2.這一部分的可以暫時(shí)不要關(guān)注。因?yàn)檫@個(gè)例子只是為了能以運(yùn)行起來(lái)的方式展示okHttp如何發(fā)起同步請(qǐng)求。
okHttp發(fā)起異步請(qǐng)求
既然android本身不支持發(fā)起同步請(qǐng)求,當(dāng)然也沒(méi)人要發(fā)起同步請(qǐng)求。這么做是能導(dǎo)致嚴(yán)重的用戶體驗(yàn)問(wèn)題。想象一下,如果你有一個(gè)瀑布流,然后瀑布流里全部顯示的都是圖片?,F(xiàn)在用戶要不斷地往下翻看瀑布流的圖片。如果這些圖片都用同步請(qǐng)求的話,什么時(shí)候可以翻一頁(yè)不說(shuō),系統(tǒng)的ANR早就跳出來(lái)了。
所以我們就探究一下如何發(fā)起一個(gè)異步的請(qǐng)求。
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.同步請(qǐng)求用execute方法,異步就用call.enqueue(new Callback()方法。
2.這個(gè)Callback接口提供了兩個(gè)方法,一個(gè)是onFailure,一個(gè)是onResponse。這兩個(gè)方法分別在請(qǐng)求失敗和成功的時(shí)候調(diào)用。
3.本來(lái)一切都似乎應(yīng)該很簡(jiǎn)單。網(wǎng)絡(luò)請(qǐng)求成功或者失敗直接在界面更新了。但是木有想到這樣會(huì)拋異常。然后看了看發(fā)現(xiàn)原來(lái)onFailure和onResponse兩個(gè)方法不是在主線程執(zhí)行。打印出來(lái)的log是:okhttp.demo.com.okhttpdemo D/###okHttp: Not Main Thread。
所以要在主線程中更新view只好想別的辦法了。在worker線程里更新主線程會(huì)拋異常。一般來(lái)說(shuō)有這么幾個(gè)方法在子線程里更新view。
在子線程更新UI線程
一、Activity的runOnUiThread方法
在Activity中有這么一個(gè)方法runOnUiThread。這個(gè)方法需要一個(gè)Runnable實(shí)例作為參數(shù)。
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方法也是一樣,扔一個(gè)Runnable的實(shí)例進(jìn)去。然后就在主線程執(zhí)行了。Toast肯定是沒(méi)有這個(gè)方法的。
MainActivity.this.mView.post(new Runnable() { public void run() { Log.d("UI thread", "I am the UI thread"); } });
三、其他
1.用Handler,這個(gè)前面的okHttp同步請(qǐng)求的例子可以用。
2.AsyncTask, 有兩個(gè)方法可以在主線程中執(zhí)行:onProgressUpdate和onPostExecute。這里我們并不是要更新進(jìn)度,所以考慮的是后一個(gè)方法。
private class BackgroundTask extends AsyncTask<String, Void, Bitmap> { protected void onPostExecute(Bitmap result) { Log.d("UI thread", "I am the UI thread"); } }
綜合以上,更新UI線程的方法里最后說(shuō)到的Handler方法和AsyncTask都太重。尤其是AsyncTask。還要繼承實(shí)現(xiàn)一堆的方法之后才可以能達(dá)到目的,同時(shí)還和我們要用的okHttp的使用方法很多不兼容的地方。
所以我們只考慮前面的兩種。但是兩種方法其實(shí)是不一樣的。當(dāng)然,這里并不是說(shuō)方法的名字不一樣。我們來(lái)看看android的源代碼,這兩個(gè)方法是如何實(shí)現(xiàn)的。
public final void runOnUiThread(Runnable action) { if (Thread.currentThread() != mUiThread) { mHandler.post(action); } else { action.run(); } } // mHandler.post(action); 之post方法的實(shí)現(xiàn) public final boolean post(Runnable r) { return sendMessageDelayed(getPostMessage(r), 0); }
方法runOnUiThread最后會(huì)調(diào)用Handler的sendMessageDelayed。但是這里只delay了0。也就是方法傳到這里的時(shí)候會(huì)立即執(zhí)行runOnUiThread的參數(shù)Runnable實(shí)例會(huì)立即執(zhí)行。
下面看看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; }
一般會(huì)執(zhí)行的是ViewRootImpl.getRunQueue().post(action);。也就是Runnable的實(shí)例只是添加到了事件隊(duì)列中,按照順序執(zhí)行。并不一定會(huì)立即執(zhí)行。
我們探討了那么多,最后就使用runOnUiThread來(lái)更新界面,也就是方法一了。
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(); } }); } });
無(wú)論請(qǐng)求成功還是失敗,都彈出一個(gè)Toast。用MainActivity.this.runOnUiThread在UI線程中彈出這個(gè)Toast。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android實(shí)現(xiàn)網(wǎng)絡(luò)圖片瀏覽器
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)網(wǎng)絡(luò)圖片瀏覽器的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05Android編程設(shè)計(jì)模式之狀態(tài)模式詳解
這篇文章主要介紹了Android編程設(shè)計(jì)模式之狀態(tài)模式,結(jié)合實(shí)例形式詳細(xì)分析了Android狀態(tài)模式的概念、功能、使用方法及相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-12-12Android studio設(shè)計(jì)簡(jiǎn)易計(jì)算器
這篇文章主要為大家詳細(xì)介紹了Android studio設(shè)計(jì)簡(jiǎn)易計(jì)算器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-04-04Android 如何攔截用戶頻繁操作(點(diǎn)擊事件)
本文主要介紹了Android 如何攔截用戶頻繁操作,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08Android實(shí)現(xiàn)炫酷的CheckBox效果
大家是不是對(duì)系統(tǒng)自帶的CheckBox產(chǎn)生乏味感了呢?今天這篇文章給大家?guī)?lái)的是一款全新的CheckBox,下面來(lái)一起看看下面的CheckBox吧!有需要的朋友們可以參考借鑒。2016-10-10實(shí)例講解Android應(yīng)用開(kāi)發(fā)中Fragment生命周期的控制
這篇文章主要介紹了Android應(yīng)用開(kāi)發(fā)中Fragment生命周期的控制,Fragment依賴于Activity,所以生命周期方面也受Activity的影響,需要的朋友可以參考下2016-02-02Android編程之短信竊聽(tīng)器實(shí)現(xiàn)方法
這篇文章主要介紹了Android編程之短信竊聽(tīng)器實(shí)現(xiàn)方法,以實(shí)例形式較為詳細(xì)的分析了Android編程實(shí)現(xiàn)竊聽(tīng)器的具體步驟與實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-11-11