Android OkHttp + Retrofit 取消請求的方法

原文連接java

前言

在某一個界面,用戶發起了一個網絡請求,由於某種緣由用戶在網絡請求完成前離開了當前界面,比較好的作法是取消這個網絡請求。對於OkHttp來講,具體是調用Callcancel方法。git

如何找到這一個網絡請求並取消掉它呢?github

OkHttp中的tag

要取消一個請求,OkHttp中能夠使用cancel方法,參考網絡

OkHttp的request對象有tag。能夠根據tag來標示請求。參考Stack Overflowide

//Set tags for your requests when you build them:
    Request request = new Request.Builder().
    url(url).tag("requestKey").build();

    //When you want to cancel:
    //A) go through the queued calls and cancel if the tag matches:
    for (Call call : mHttpClient.dispatcher().queuedCalls()) {
        if (call.request().tag().equals("requestKey"))
            call.cancel();
    }

    //B) go through the running calls and cancel if the tag matches:
    for (Call call : mHttpClient.dispatcher().runningCalls()) {
        if (call.request().tag().equals("requestKey"))
            call.cancel();
    }

複製代碼

Retrofit中並無顯示地提供取消請求的接口。2018年時Retrofit仍未提供直接訪問call對象的方法 那麼如何找到目標網絡請求呢?ui

Retrofit加入自定義header

給每一個與頁面(Activity,Fragment)相關的request加入自定義header,參考。 給OkHttpClient添加攔截器。標記出頁面的生存狀態。若是頁面銷燬了,則取消對應的request。url

以GithubOnAndroid項目爲例,github.com/RustFisher/…spa

添加標記

持有一個ConcurrentHashMap<String, Boolean>來標記頁面存活狀態。code

private static ConcurrentHashMap<String, Boolean> actLiveMap = new ConcurrentHashMap<>(); // 標記Activity是否存活

    public static void markPageAlive(String actName) {
        actLiveMap.put(actName, true);
    }

    public static void markPageDestroy(String actName) {
        actLiveMap.put(actName, false);
    }
複製代碼

Activity中登記界面狀態

給當前Activity起名字。每一個Activity的標記名必須惟一。對象

private static final String MY_ACT_NAME = "xxx1Activity";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        NetworkCenter.markPageAlive(MY_ACT_NAME);
        // ...
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        NetworkCenter.markPageDestroy(MY_ACT_NAME);
        // ...
    }
複製代碼

OkHttpClient添加攔截器

給OkHttpClient添加攔截器,在攔截器中檢查頁面的存活狀況。 檢查後,把這個自定義header移除掉。

public static final String HEADER_ACT_NAME = "Activity-Name"; // 標記Activity界面名字

    private Interceptor lifeInterceptor = new Interceptor() {
        @Override
        public Response intercept(Chain chain) throws IOException {
            Request request = chain.request();
            String actName = request.header(HEADER_ACT_NAME);
            if (!TextUtils.isEmpty(actName)) {
                Log.d(TAG, "lifeInterceptor: actName: " + actName);
                Boolean actLive = actLiveMap.get(actName);
                if (actLive == null || !actLive) {
                    chain.call().cancel();
                    Log.d(TAG, "lifeInterceptor: 取消請求, actName: " + actName);
                } else {
                    Log.d(TAG, "lifeInterceptor: 發起請求, actName: " + actName);
                }
            }
            Request newRequest = request.newBuilder().removeHeader(HEADER_ACT_NAME).build();
            return chain.proceed(newRequest);
        }
    };


OkHttpClient = new OkHttpClient.Builder()
        .readTimeout(10, TimeUnit.SECONDS)
        .connectTimeout(10, TimeUnit.SECONDS)
        .addInterceptor(lifeInterceptor) // 添加攔截器
        .build();
複製代碼

call.cancel()後,不會再走Retrofit的subscribe方法。

添加header

@GET("users/{owner}/repos")
    Observable<List<UserRepo>> userRepo(
            @Header(NetworkCenter.HEADER_ACT_NAME) @Nullable String actName,
            @Path("owner") String owner,
            @Query("sort") String sortType);
複製代碼

相關連接 rustfisher.com/2019/06/26/…

相關文章
相關標籤/搜索