對于搜索頁面“熱門搜索網(wǎng)絡請求”的總結(jié)

網(wǎng)絡請求

歷史記錄以及熱門搜索

整個過程主要涉及4個類,(SearchActivity僅僅起到顯示作用所以不算)猪杭,至于Adaptor中數(shù)據(jù)添加不放在此時總結(jié)之中。

熱搜網(wǎng)絡總體思路

Fragment(activity)中調(diào)用viewmodel聲明的livedata的獲取方法妥衣,然后添加observer皂吮,在響應方法中進行adaptor的數(shù)據(jù)添加,(observer的響應方法會將livedata的數(shù)據(jù)作為參數(shù)傳遞)税手。在viewmodel中用addSource實現(xiàn)livedata之間的拼接和方法響應問題涮较,然而在實際應用中的這個livedata不是原生的,是自定義livedata冈止,在livedata的生命周期中進行一些操作狂票,例如自動觸發(fā)請求并進行數(shù)據(jù)處理。在SearchSuggestFragment中調(diào)用SearchSuggestViewmodel的getHistoryAndRank,返回的事一個LiveData熙暴,但是在這個方法中闺属,為兩個中轉(zhuǎn)livedata創(chuàng)造了分別的獲取方法和響應者processHistoryAndRank()來負責最終livedata的拼接,拼接之后Fragment中的observer才會響應并賦值給adapter周霉。

類圖
調(diào)用流程

其實真正實現(xiàn)請求的在JceRequestLiveData中掂器,fireRequest方法在onActive()時就會觸發(fā),至于請求的url俱箱、對請求回來的數(shù)據(jù)進行解析和包裝還有成功和失敗的響應方法都可以由針對不同需求的子類來實現(xiàn)国瓮。

下面是具體過程的代碼。

SearchSuggestionFragment

 ②
 @NonNull
@Override
public ViewGroup onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState)
{
  
    final SearchSuggestionViewModel model = ViewModelProviders.of(this) // 生命周期跟隨Fragment
                                                              .get(SearchSuggestionViewModel.class);
    model.setRequestingSource(
            SearchSuggestionViewModel.getKeywordRequestLoading(activity),
            SearchSuggestionViewModel.getKeywordInputEmpty(activity));

    model.getSearchResult().observe(this, this::setSearchResult);
    model.getHistoryAndRank(mOpenSearchFromFrameType, mOpenSearchFromId).observe(this, this::setHistoryAndRank);
    SearchSuggestionViewModel.getSuggestion(activity, mOpenSearchFromFrameType, mOpenSearchFromId).observe(this, this::setSuggestion);
    mSearchViewModel.getSearchError().observe(this, this::setSearchError);
    return view;
}

11
@MainThread
private void setHistoryAndRank(@Nullable List<RowItem> list)
{
    mHistoryAndRank = list;
    final ViewSearchSuggestionBinding binding = mBinding;
    if (binding != null)
    {
        final AsyncListVMAdapter<RowItem> historyAndRankAdapter = getHistoryAndRankAdapter();
        historyAndRankAdapter.submitList(list);
        final boolean hasFocus = binding.getRoot().hasFocus();
        setSearchResult(mSearchResult);
        if (hasFocus && !binding.getRoot().hasFocus())
        {
            binding.getRoot().requestFocus();
        }
    }
}

SearchSuggestionViewModel

③
@NonNull
@MainThread
public LiveData<List<RowItem>> getHistoryAndRank(String openSearchFromFrameType, String openSearchFromId)
{
    if (mHistoryAndRankLiveData == null)
    {
        final MediatorLiveData<List<RowItem>> data = new MediatorLiveData<>();
        data.addSource(LocalSearchHistoryManager.getInstance(), this::onLocalHistoryChanged);
        data.addSource(new SearchRankLiveData(openSearchFromFrameType, openSearchFromId), this::onRankListChanged);
        mHistoryAndRankLiveData = data;
        processHistoryAndRank();
    }
    return mHistoryAndRankLiveData;
}

⑩
 @MainThread
private void onRankListChanged(@Nullable List<RowItem> rankList)
{
    mRankRowItems = rankList;
    processHistoryAndRank();
}
@MainThread
private void processHistoryAndRank()
{
    if (mHistoryAndRankRowItems != null && !mHistoryAndRankRowItems.isEmpty())
    {
        // 給到UI的數(shù)據(jù)狞谱,每一份都應該是獨立的乃摹,千萬不要隨便復用對象,容易產(chǎn)生"不明原因"的奇怪現(xiàn)象
        mHistoryAndRankRowItems = new ArrayList<>(mHistoryAndRankRowItems.size());
    }
    else
    {
        mHistoryAndRankRowItems = new ArrayList<>();
    }


    if (mRankRowItems != null && !mRankRowItems.isEmpty())
    {
        if (getLocalHistoryPosition() == 0)
        {
            if (mLocalHistoryRowItems != null && !mLocalHistoryRowItems.isEmpty())
            {
                mHistoryAndRankRowItems.addAll(mLocalHistoryRowItems);
            }
            if (mRankRowItems != null && !mRankRowItems.isEmpty())
            {
                mHistoryAndRankRowItems.addAll(mRankRowItems);
            }
        }
        else
        {
            if (mRankRowItems != null && !mRankRowItems.isEmpty())
            {
                mHistoryAndRankRowItems.addAll(mRankRowItems);
            }
            if (mLocalHistoryRowItems != null && !mLocalHistoryRowItems.isEmpty())
            {
                mHistoryAndRankRowItems.addAll(mLocalHistoryRowItems);
            }
        }
    }


    if (mHistoryAndRankLiveData != null)
    {
        mHistoryAndRankLiveData.postValue(Collections.unmodifiableList(mHistoryAndRankRowItems));
    }
    else
    {
        TVCommonLog.w(TAG, "processHistoryAndRank: missing live data");
    }
}

SearchRankLiveData

④
 public SearchRankLiveData(@NonNull String openSearchFromFrameType, @NonNull String openSearchFromId)
{
    mUrl = UrlConstants.CGIPrefix.URL_SEARCH_RANK
            + "&req_size=10"
            + "&raw=1" +  "&" + TenVideoGlobal.getCommonUrlSuffix();
}
 ⑦
 @Override
protected String makeRequestUrl()
{
    return mUrl;
}
⑧
 @Override
protected List<RowItem> parseJce(byte[] bytes) throws JceDecodeException
{
    VideoPageRspV2 rsp = null;
    try
    {
        rsp = new JceCommonConvertor<>(VideoPageRspV2.class).convertBytes2JceStruct(bytes);
    }
    catch (BufferUnderflowException e)
    {
        TVCommonLog.w(TAG, "parseJce: BufferUnderflowException");
        return null;
    }
    if (rsp == null)
    {
        TVCommonLog.w(TAG, "parseJce: fail to parse jce");
        return null;
    }
    if (rsp.result != null && rsp.result.ret != RetCode._SUCCESS)
    {
        TVCommonLog.w(TAG, "parseJce: ret = [" + rsp.result.ret + "], msg = [" + rsp.result.msg + "]");
        return null;
    }
    // 到此跟衅,算是請求成功了孵睬。下面,需要再做數(shù)據(jù)轉(zhuǎn)換伶跷,不清楚到底有沒有數(shù)據(jù)
    final ListData data = rsp.data;
    final ArrayList<RowItem> ret = new ArrayList<>();
   .....
}

JceRequestLiveData

 ⑤
@Override
protected void onActive()
{
    super.onActive();
    if (mLatestRequest == null && getValue() == null)
    {
        // 自動請求數(shù)據(jù)
        fireRequest(false);
    }
}
@Override
protected void onInactive()
{
    super.onInactive();
    if (mLatestRequest != null && !mForcingRequest)
    {
       //當LiveData感知的view的生命周期結(jié)束的時候取消請求
        cancelRequest();
    }
}
⑥
 private void fireRequest(boolean force)
{
    mForcingRequest = force;
    mLatestRequest = new BaseJceRequest<T>()
    {
        @Override
        protected String makeRequestUrl()
        {
            return JceRequestLiveData.this.makeRequestUrl();
        }

        @Override
        public String getRequstName()
        {
            return JceRequestLiveData.this.getRequestName();
        }

        @Override
        public T parseJce(byte[] bytes) throws JceDecodeException
        {
            return JceRequestLiveData.this.parseJce(bytes);
        }
    };
    if (getValue() == null)
    {
        mLatestRequest.setRequestMode(mRequestMode);
    }
    else
    {
        mLatestRequest.setRequestMode(Request.LoadMode.SERVER);
    }
    GlobalManager.getInstance().getAppEngine().get(mLatestRequest, mResponseHandler);
}
⑨
private final AppResponseHandler<T> mResponseHandler = new AppResponseHandler<T>()
{
    @Override
    public void onSuccess(T data, boolean fromCache)
    {
        final T legalData = JceRequestLiveData.this.onSuccess(data, fromCache);
        if (legalData == data)
        {
            postValue(data);
        }
    }

    @Override
    public void onFailure(RespErrorData error)
    {
        JceRequestLiveData.this.onFailure(error);
    }
};
 protected abstract T parseJce(byte[] bytes) throws JceDecodeException;

搜索歷史記錄的本地獲取


在上述圖中的getHistoryAndRank中就addSourse兩個LiveData掰读,

  data.addSource(LocalSearchHistoryManager.getInstance(), this::onLocalHistoryChanged);
  data.addSource(new SearchRankLiveData(openSearchFromFrameType, openSearchFromId), this::onRankListChanged);

其中的SearchRankLiveData是用于獲取熱門搜索的LiveData,而LocalSearchHistoryManager就是從內(nèi)存中獲取搜索歷史的LiveData叭莫,實際上這個單例是軟引用的蹈集,

解釋一下各種引用:

⑴強引用(StrongReference)
強引用是使用最普遍的引用。如果一個對象具有強引用雇初,那垃圾回收器絕不會回收它拢肆。當內(nèi)存空間不足,Java虛擬機寧愿拋出OutOfMemoryError錯誤,使程序異常終止善榛,也不會靠隨意回收具有強引用的對象來解決內(nèi)存不足的問題辩蛋。

⑵軟引用(SoftReference)
如果一個對象只具有軟引用,則內(nèi)存空間足夠移盆,垃圾回收器就不會回收它悼院;如果內(nèi)存空間不足了,就會回收這些對象的內(nèi)存咒循。只要垃圾回收器沒有回收它据途,該對象就可以被程序使用。軟引用可用來實現(xiàn)內(nèi)存敏感的高速緩存叙甸。
軟引用可以和一個引用隊列(ReferenceQueue)聯(lián)合使用颖医,如果軟引用所引用的對象被垃圾回收器回收,Java虛擬機就會把這個軟引用加入到與之關聯(lián)的引用隊列中裆蒸。

⑶弱引用(WeakReference)
弱引用與軟引用的區(qū)別在于:只具有弱引用的對象擁有更短暫的生命周期熔萧。在垃圾回收器線程掃描它所管轄的內(nèi)存區(qū)域的過程中,一旦發(fā)現(xiàn)了只具有弱引用的對象僚祷,不管當前內(nèi)存空間足夠與否佛致,都會回收它的內(nèi)存。不過辙谜,由于垃圾回收器是一個優(yōu)先級很低的線程俺榆,因此不一定會很快發(fā)現(xiàn)那些只具有弱引用的對象。
弱引用可以和一個引用隊列(ReferenceQueue)聯(lián)合使用装哆,如果弱引用所引用的對象被垃圾回收罐脊,Java虛擬機就會把這個弱引用加入到與之關聯(lián)的引用隊列中。

⑷虛引用(PhantomReference)
“虛引用”顧名思義蜕琴,就是形同虛設萍桌,與其他幾種引用都不同,虛引用并不會決定對象的生命周期奸绷。如果一個對象僅持有虛引用梗夸,那么它就和沒有任何引用一樣,在任何時候都可能被垃圾回收器回收号醉。
虛引用主要用來跟蹤對象被垃圾回收器回收的活動。虛引用與軟引用和弱引用的一個區(qū)別在于:虛引用必須和引用隊列 (ReferenceQueue)聯(lián)合使用辛块。當垃圾回收器準備回收一個對象時畔派,如果發(fā)現(xiàn)它還有虛引用,就會在回收對象的內(nèi)存之前润绵,把這個虛引用加入到與之 關聯(lián)的引用隊列中线椰。

具體獲取數(shù)據(jù)庫中信息的代碼為以下:(DBQueryRequest是封裝的數(shù)據(jù)庫查詢類,keys是一個含有一個string類型標記值的數(shù)組)

  DBQueryRequest<DataPair> request = new DBQueryRequest<DataPair>();
  request.setTableName(LocalCacheConstract.LocalCaches.TABLE_NAME);
  request.setSelection(DatabaseUtils.getSelection(LocalCacheConstract.LocalCaches.ID, keys));
  ArrayList<DataPair> pairs = request.sendRequestSync();

為什么這么寫呢尘盼?

其實在類圖和流程圖中個各類的分工已經(jīng)很明確憨愉,完全符合MVVM架構(gòu)結(jié)合liveData的實現(xiàn)方式烦绳,

  • View(activity、fragment)
    負責了數(shù)據(jù)在數(shù)據(jù)出口輸出時響應回調(diào)配紫,進行數(shù)據(jù)刷新径密,
  • ViewModel
    負責實例化各種存儲數(shù)據(jù)的容器,即liveData,包括各個liveData之間的數(shù)據(jù)傳值和包裝邏輯
  • Model用自定義LiveData代替
    實現(xiàn)數(shù)據(jù)請求等操作

其實主要是想利用livedata的生命周期躺孝,從而實現(xiàn)數(shù)據(jù)的自動傳遞享扔。

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市植袍,隨后出現(xiàn)的幾起案子惧眠,更是在濱河造成了極大的恐慌,老刑警劉巖于个,帶你破解...
    沈念sama閱讀 207,248評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件氛魁,死亡現(xiàn)場離奇詭異,居然都是意外死亡厅篓,警方通過查閱死者的電腦和手機秀存,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評論 2 381
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來贷笛,“玉大人应又,你說我怎么就攤上這事》啵” “怎么了株扛?”我有些...
    開封第一講書人閱讀 153,443評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長汇荐。 經(jīng)常有香客問我洞就,道長,這世上最難降的妖魔是什么掀淘? 我笑而不...
    開封第一講書人閱讀 55,475評論 1 279
  • 正文 為了忘掉前任旬蟋,我火速辦了婚禮,結(jié)果婚禮上革娄,老公的妹妹穿的比我還像新娘倾贰。我一直安慰自己,他們只是感情好拦惋,可當我...
    茶點故事閱讀 64,458評論 5 374
  • 文/花漫 我一把揭開白布匆浙。 她就那樣靜靜地躺著,像睡著了一般厕妖。 火紅的嫁衣襯著肌膚如雪首尼。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,185評論 1 284
  • 那天,我揣著相機與錄音软能,去河邊找鬼迎捺。 笑死,一個胖子當著我的面吹牛查排,可吹牛的內(nèi)容都是我干的凳枝。 我是一名探鬼主播,決...
    沈念sama閱讀 38,451評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼雹嗦,長吁一口氣:“原來是場噩夢啊……” “哼范舀!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起了罪,我...
    開封第一講書人閱讀 37,112評論 0 261
  • 序言:老撾萬榮一對情侶失蹤锭环,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后泊藕,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體辅辩,經(jīng)...
    沈念sama閱讀 43,609評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,083評論 2 325
  • 正文 我和宋清朗相戀三年娃圆,在試婚紗的時候發(fā)現(xiàn)自己被綠了玫锋。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,163評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡讼呢,死狀恐怖撩鹿,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情悦屏,我是刑警寧澤节沦,帶...
    沈念sama閱讀 33,803評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站础爬,受9級特大地震影響甫贯,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜看蚜,卻給世界環(huán)境...
    茶點故事閱讀 39,357評論 3 307
  • 文/蒙蒙 一叫搁、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧供炎,春花似錦渴逻、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至纽竣,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蜓氨。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評論 1 261
  • 我被黑心中介騙來泰國打工聋袋, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人穴吹。 一個月前我還...
    沈念sama閱讀 45,636評論 2 355
  • 正文 我出身青樓幽勒,卻偏偏與公主長得像,于是被迫代替她去往敵國和親港令。 傳聞我的和親對象是個殘疾皇子啥容,可洞房花燭夜當晚...
    茶點故事閱讀 42,925評論 2 344

推薦閱讀更多精彩內(nèi)容

  • 懶得處理樣式了, 將就著看吧. 官網(wǎng)地址: https://developer.android.com/topic...
    Reddington_604e閱讀 1,645評論 0 1
  • ??LiveData是一個可被觀察的數(shù)據(jù)持有者類。與常規(guī)的Observable不同顷霹,LiveData能意識到應用程...
    鶴鶴閱讀 47,004評論 6 42
  • LiveData是一種可觀察的數(shù)據(jù)存儲器類咪惠。與常規(guī)的可觀察類不同,LiveData 具有生命周期感知能力淋淀,意指它遵...
    tse1y閱讀 3,010評論 0 2
  • [TOC] LiveData LiveData概述 LiveData是一個可觀察的數(shù)據(jù)持有者類遥昧。與常規(guī)observ...
    雪晨杰閱讀 681評論 0 0
  • 其(一) 【七絕】訪友不遇 和鑿冰而釣 門鎖...
    大康詩文閱讀 1,550評論 39 40