okhttp處理短時間內(nèi)的重復(fù)請求

關(guān)鍵點

  • 判斷請求是否相同: request轉(zhuǎn)key,根據(jù)key判斷
  • 請求進行中: 等待
  • 請求已完成: 返回緩存的response
  • 過去這段時間后,清除緩存的response

代碼實現(xiàn):

public class SameRequestFilterInterceptor implements Interceptor {

    static WeakHashMap<String, ResonseForClone> responseWeakHashMap = new WeakHashMap<>();
    static WeakHashMap<String, WeakReference<Call>> calls = new WeakHashMap<>();

    static Charset UTF_8 = Charset.forName("UTF-8");
    static IConfig config;
    static boolean enableFilter;
    static boolean debug;
    static Handler handler;

    /**
     * @param enableFilter 全局開關(guān)
     * @param config       配置信息
     */
    public static void config(boolean debug, boolean enableFilter, IConfig config) {
        SameRequestFilterInterceptor.enableFilter = enableFilter;
        SameRequestFilterInterceptor.config = config;
        SameRequestFilterInterceptor.debug = debug;
    }


    @Override
    public Response intercept(Chain chain) throws IOException {
        Request request = chain.request();
        if (!enableFilter) {
            return chain.proceed(request);
        }

        if (config == null) {
            return chain.proceed(request);
        }
        if (!config.shouldFilter(request.url().toString())) {
            return chain.proceed(request);
        }

        String key = generateKey(request);

        return check(chain, request, key);
    }

    private Response check(Chain chain, Request request, String key) throws IOException {
        try {
            //從緩存的call和response中判斷要不要等待
            boolean needwait = needwait(key);

            if (!needwait) {
                if (responseWeakHashMap.containsKey(key)) {
                    return responseWeakHashMap.get(key).getClonedResonse();
                } else {
                    //直接執(zhí)行請求:
                    //將response緩存起來
                    return realExceute(chain, request, key);
                }
            } else {
                Thread.sleep(2000);
                return check(chain, request, key);
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
            if (responseWeakHashMap.containsKey(key)) {
                return responseWeakHashMap.get(key).getClonedResonse();
            } else {
                //直接執(zhí)行請求:
                //將response緩存起來
                return realExceute(chain, request, key);
            }
        }
    }

    @NonNull
    private Response realExceute(Chain chain, Request request, String key) throws IOException {
        calls.put(key, new WeakReference<>(chain.call()));
        Response response = chain.proceed(request);


        if (response.isSuccessful() && response.body() != null) {
            ResponseBody responseBody = response.body();
            BufferedSource source = responseBody.source();
            source.request(responseBody.contentLength() > 0 ? responseBody.contentLength() : Integer.MAX_VALUE);
            //嚇人?
            Buffer buffer = source.buffer();
            Charset charset = UTF_8;
            MediaType contentType = responseBody.contentType();
            if (contentType != null) {
                charset = contentType.charset(UTF_8);
            }
            String bodyString = buffer.clone().readString(charset);

            ResponseBody cloneBody = ResponseBody.create(response.body().contentType(), bodyString);

            Response responseClone = new Response.Builder()
                    .code(response.code())
                    .protocol(response.protocol())
                    .message(response.message())
                    .body(cloneBody)
                    .headers(response.headers())
                    .header("cachedResonse", "yes")
                    .request(request)
                    .build();
            responseWeakHashMap.put(key, new ResonseForClone(bodyString, responseClone));
            calls.remove(key);
            getMainHandler().postDelayed(new Runnable() {
                @Override
                public void run() {
                    //1min后移除緩存的response:
                    responseWeakHashMap.remove(key);
                    logw(config.responseCacheTimeInMills() + "時間到了,清除緩存的response");
                }
            }, config.responseCacheTimeInMills());

        }

        return response;
    }

    private static Handler getMainHandler() {
        if (handler == null) {
            handler = new Handler(Looper.getMainLooper());
        }
        return handler;
    }

    private boolean needwait(String key) {
        if (responseWeakHashMap.containsKey(key)) {
            logw("有緩存的response,直接去讀緩存,并組裝新的response");
            return false;
        }
        if (calls.containsKey(key)) {
            WeakReference<Call> callWeakReference = calls.get(key);
            if (callWeakReference == null) {
                logw("不需要等待,直接發(fā)請求 call WeakReference not exist:");
                return false;
            }
            Call call = callWeakReference.get();
            if (call == null || call.isCanceled()) {
                logw("不需要等待,直接發(fā)請求 call not exist or is canceld:" + call);
                return false;
            }
            logw("請求可能正在等待或正在執(zhí)行-needwait call is running:" + call);
            //請求可能正在等待或正在執(zhí)行
            return true;
        }
        logw("任何地方都沒有,不需要等,直接執(zhí)行請求");
        //任何地方都沒有,不需要等,直接執(zhí)行請求
        return false;
    }

    private static void logw(String str) {
        if (debug) {
            Log.w("requestFilter", str);
        }
    }

    /**
     * @param request
     * @return
     */
    private String generateKey(Request request) {

        return config.generateCacheKey(request);
    }

    class ResonseForClone {
        String body;
        Response response;

        public ResonseForClone(String body, Response response) {
            this.body = body;
            this.response = response;
        }

        public Response getClonedResonse() {
            ResponseBody cloneBody = ResponseBody.create(response.body().contentType(), body);
            Response responseClone = new Response.Builder()
                    .code(response.code())
                    .protocol(response.protocol())
                    .message(response.message())
                    .body(cloneBody)
                    .headers(response.headers())
                    .header("cachedResonse", "yes")
                    .request(response.request())
                    .build();
            return responseClone;
        }
    }

    public interface IConfig {
        boolean shouldFilter(String url);

        String generateCacheKey(Request request);

        long responseCacheTimeInMills();
    }
}

使用:

SameRequestFilterInterceptor.config(Config.enableLog(), true, new SameRequestFilterInterceptor.IConfig() {
            @Override
            public boolean shouldFilter(String url) {
                return true;
            }

            @Override
            public String generateCacheKey(Request request) {
                return request.url().toString();
            }

            @Override
            public long responseCacheTimeInMills() {
                return 60000;
            }
        });

commonInterceptors.add(new SameRequestFilterInterceptor());

demo:

for (int i = 0; i < 20; i++) {
            request();
        }

日志:


image.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌搂蜓,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,372評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件冯凹,死亡現(xiàn)場離奇詭異躁倒,居然都是意外死亡撰豺,警方通過查閱死者的電腦和手機琳疏,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評論 3 392
  • 文/潘曉璐 我一進店門有决,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人空盼,你說我怎么就攤上這事书幕。” “怎么了揽趾?”我有些...
    開封第一講書人閱讀 162,415評論 0 353
  • 文/不壞的土叔 我叫張陵台汇,是天一觀的道長。 經(jīng)常有香客問我篱瞎,道長苟呐,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,157評論 1 292
  • 正文 為了忘掉前任奔缠,我火速辦了婚禮掠抬,結(jié)果婚禮上吼野,老公的妹妹穿的比我還像新娘校哎。我一直安慰自己,他們只是感情好瞳步,可當(dāng)我...
    茶點故事閱讀 67,171評論 6 388
  • 文/花漫 我一把揭開白布闷哆。 她就那樣靜靜地躺著,像睡著了一般单起。 火紅的嫁衣襯著肌膚如雪抱怔。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,125評論 1 297
  • 那天嘀倒,我揣著相機與錄音屈留,去河邊找鬼局冰。 笑死,一個胖子當(dāng)著我的面吹牛灌危,可吹牛的內(nèi)容都是我干的康二。 我是一名探鬼主播,決...
    沈念sama閱讀 40,028評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼勇蝙,長吁一口氣:“原來是場噩夢啊……” “哼沫勿!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起味混,我...
    開封第一講書人閱讀 38,887評論 0 274
  • 序言:老撾萬榮一對情侶失蹤产雹,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后翁锡,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蔓挖,經(jīng)...
    沈念sama閱讀 45,310評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,533評論 2 332
  • 正文 我和宋清朗相戀三年馆衔,在試婚紗的時候發(fā)現(xiàn)自己被綠了时甚。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,690評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡哈踱,死狀恐怖荒适,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情开镣,我是刑警寧澤刀诬,帶...
    沈念sama閱讀 35,411評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站邪财,受9級特大地震影響陕壹,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜树埠,卻給世界環(huán)境...
    茶點故事閱讀 41,004評論 3 325
  • 文/蒙蒙 一糠馆、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧怎憋,春花似錦又碌、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至癌别,卻和暖如春皂岔,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背展姐。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評論 1 268
  • 我被黑心中介騙來泰國打工躁垛, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留剖毯,地道東北人。 一個月前我還...
    沈念sama閱讀 47,693評論 2 368
  • 正文 我出身青樓教馆,卻偏偏與公主長得像速兔,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子活玲,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,577評論 2 353