OkHttp基礎(chǔ)學習(一),Get請求

學習資料:

打算花段時間,系統(tǒng)性的學習下OkHttp相關(guān)知識。計劃先從最簡單的get請求學起


1. Get請求

運行結(jié)果

網(wǎng)絡(luò)請求接口來自易源-美女圖片


1.1 GetActivity代碼

布局:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/activity_get"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <android.support.v4.widget.NestedScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent">

        <TextView
            android:id="@+id/get_activity_tv_show"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:textAllCaps="false"
            android:textColor="@color/colorAccent"
            android:textSize="15sp" />
    </android.support.v4.widget.NestedScrollView>

</RelativeLayout>

直接將TextView放在一個NestScrollView


OKHttp使用需要3個對象:

  1. OkHttpClient
  2. Request
  3. Call

Activity代碼

public class GetActivity extends AppCompatActivity implements ResultCallback<String> {
    private TextView tv_info;
    private Platform mPlatform;//平臺對象,內(nèi)部最終調(diào)用 , handler.post(runnable)

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_get);
        //得到 Platform  對象
        mPlatform = Platform.get();
        initView();
    }

    private void initView() {
        tv_info = (TextView) findViewById(R.id.get_activity_tv_show);
        request();
    }

    /**
     * 請求網(wǎng)絡(luò)
     */
    private void request() {
        //創(chuàng)建OkHttpClient 設(shè)置超時時間為10s
        OkHttpClient okHttpClient = new OkHttpClient.Builder().connectTimeout(10, TimeUnit.SECONDS).build();
        //創(chuàng)建Request
        Request request = new Request.Builder().get().url(Urls.GET_URL).build();
        //創(chuàng)建Call
        Call call = okHttpClient.newCall(request);
        //異步執(zhí)行請求
        call.enqueue(new Callback() {
            @Override
            public void onFailure(Call call, IOException e) {
                sendFailResultCallback(e);//失敗回調(diào)
            }

            @Override
            public void onResponse(Call call, Response response) throws IOException {
                //請求結(jié)果
                ResponseBody responseBody = null;
                try {
                    //判斷請求是否取消
                    if (call.isCanceled()) {
                        sendFailResultCallback(new IOException("Request Canceled"));
                        return;
                    }
                    //獲取請求結(jié)果 ResponseBody
                    responseBody = response.body();
                    //獲取字符串
                    String json = responseBody.string();
                    Log.e("activity", json);
                    //成功回調(diào)
                    sendSuccessResultCallback(json);
                } catch (Exception e) {//發(fā)生異常,失敗回調(diào)
                    sendFailResultCallback(e);
                } finally {//記得關(guān)閉操作
                    if (null != responseBody) {
                        responseBody.close();
                    }
                }
            }
        });
    }

    /**
     * 失敗回調(diào)接口方法
     */

    @Override
    public void sendFailResultCallback(final Exception e) {
        mPlatform.execute(new Runnable() {
            @Override
            public void run() {
                String info = "Fail Message --> " + e.getMessage();
                tv_info.setText(info);
            }
        });
    }

    /**
     * 成功回調(diào)方法
     */

    @Override
    public void sendSuccessResultCallback(final String result) {
        mPlatform.execute(new Runnable() {
            @Override
            public void run() {
                tv_info.setText(JsonFormatUtils.formatJson(result));
            }
        });
    }
}

call.execute()是同步請求,主線程中無法進行網(wǎng)絡(luò)請求,需要使用異步call.enqueue()方法

由于網(wǎng)絡(luò)請求在子線程中,不能更新UI,考慮結(jié)合使用Handler,GetActivity中,并沒有直接創(chuàng)建出來一個Handler來使用,而是利用了一個Platform對象,在onCreat()方法中,mPlatform = Platform.get()


1.2 ResultCallback

不明白為啥一開始就想到先寫個接口。。。寫了就寫了吧

public interface ResultCallback <T>{
    //失敗時
    void sendFailResultCallback(Exception e);
    //成功時
    void sendSuccessResultCallback(T t);
}

1.3 Platform

之前看鴻洋大神的Android 一個改善的okHttp封裝庫,在代碼里面看到的這個類

代碼中有句注釋 Copyright (C) 2013 Square, Inc.,說明源自Square公司,在Retrofit源碼中也有用到,就學習下怎么使用

Platform代碼

 public class Platform {
    private static final Platform PLATFORM = findPlatform();

    public static Platform get() {
        return PLATFORM;
    }

    private static Platform findPlatform() {
        try {
            Class.forName("android.os.Build");
            if (Build.VERSION.SDK_INT != 0) {
                return new Android();
            }
        } catch (ClassNotFoundException ignored) {
            ignored.printStackTrace();
        }
        return new Platform();
    }

    public Executor defaultCallbackExecutor() {
        return Executors.newCachedThreadPool();
    }

    public void execute(Runnable runnable) {
        defaultCallbackExecutor().execute(runnable);
    }


    static class Android extends Platform {
        @Override
        public Executor defaultCallbackExecutor() {
            return new MainThreadExecutor();
        }

        static class MainThreadExecutor implements Executor {
            private final Handler handler = new Handler(Looper.getMainLooper());

            @Override
            public void execute(Runnable r) {
                handler.post(r);
            }
        }
    }
}

使用很簡單

 mPlatform.execute(new Runnable() {
      //UI線程
      do something 。。。
 }

這段代碼感覺可以拿來加深理解 多態(tài)


mPlatform = Platform.get(),在findPlatform()方法中,正常情況,返回的new Android()對象,實際是通過new Android(),得到了Platform的子類Android的一個對象

之后的mPlatform.execute(),其實就是子類Android直接調(diào)用父類Platform中的execute()方法

然而在execute()方法中,調(diào)用了defaultCallbackExecutor()

Platform中的defaultCallbackExecutor()方法:

public Executor defaultCallbackExecutor() {
    return Executors.newCachedThreadPool();
}

通過newCachedThreadPool(),創(chuàng)建出一個默認的緩存線程池


子類Android中,重寫了defaultCallbackExecutor()方法

    @Override
   public Executor defaultCallbackExecutor() {
       return new MainThreadExecutor();
   }

自定義了一個MainThreadExecutor,內(nèi)部便是對Handler的回調(diào),這個Handler是通過new Handler(Looper.getMainLooper())創(chuàng)建出來的一個UI主線程的Handler,也就可以直接用來更新UI

mPlatform.execute(Runnbale)正常情況下,最終會被Handler回調(diào)

這段代碼的質(zhì)量明顯比上面GetActivity的代碼高N條街,Square公司的,不高才怪。GetActivity中,我多此一舉的寫了個接口

多學習,以后我也要敲出這樣高質(zhì)量的代碼,哈哈 :)


2. 最后

最簡單的異步get請求,下篇接著學習post請求,文件上傳下載之類的

有錯誤請指出

共勉 :)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 178,725評論 25 709
  • afinalAfinal是一個android的ioc,orm框架 https://github.com/yangf...
    passiontim閱讀 15,835評論 2 45
  • 參考Android網(wǎng)絡(luò)請求心路歷程Android Http接地氣網(wǎng)絡(luò)請求(HttpURLConnection) 一...
    合肥黑閱讀 21,696評論 7 63
  • 我已忘記什么時候向往遠方了。小時候,遠方就是長大,能做自己想做的事;中學時,遠方就窗外,不清楚要去哪里,但總之不在...
    姜子潺閱讀 337評論 2 0
  • 一、軟文營銷,價值為王,任何不以提供價值為手段的軟文營銷都是耍流氓 二、人對信息的感知過程: 1、選擇性注意 1)...
    踏雁尋花閱讀 642評論 0 11

友情鏈接更多精彩內(nèi)容