对于搜索页面“热门搜索网络请求”的总结

网络请求

历史记录以及热门搜索

整个过程主要涉及4个类,(SearchActivity仅仅起到显示作用所以不算),至于Adaptor中数据添加不放在此时总结之中。

热搜网络总体思路

Fragment(activity)中调用viewmodel声明的livedata的获取方法,然后添加observer,在响应方法中进行adaptor的数据添加,(observer的响应方法会将livedata的数据作为参数传递)。在viewmodel中用addSource实现livedata之间的拼接和方法响应问题,然而在实际应用中的这个livedata不是原生的,是自定义livedata,在livedata的生命周期中进行一些操作,例如自动触发请求并进行数据处理。在SearchSuggestFragment中调用SearchSuggestViewmodel的getHistoryAndRank,返回的事一个LiveData,但是在这个方法中,为两个中转livedata创造了分别的获取方法和响应者processHistoryAndRank()来负责最终livedata的拼接,拼接之后Fragment中的observer才会响应并赋值给adapter。

类图
调用流程

其实真正实现请求的在JceRequestLiveData中,fireRequest方法在onActive()时就会触发,至于请求的url、对请求回来的数据进行解析和包装还有成功和失败的响应方法都可以由针对不同需求的子类来实现。

下面是具体过程的代码。

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的数据,每一份都应该是独立的,千万不要随便复用对象,容易产生"不明原因"的奇怪现象
        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;
    }
    // 到此,算是请求成功了。下面,需要再做数据转换,不清楚到底有没有数据
    final ListData data = rsp.data;
    final ArrayList<RowItem> ret = new ArrayList<>();
   .....
}

JceRequestLiveData

 ⑤
@Override
protected void onActive()
{
    super.onActive();
    if (mLatestRequest == null && getValue() == null)
    {
        // 自动请求数据
        fireRequest(false);
    }
}
@Override
protected void onInactive()
{
    super.onInactive();
    if (mLatestRequest != null && !mForcingRequest)
    {
       //当LiveData感知的view的生命周期结束的时候取消请求
        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就是从内存中获取搜索历史的LiveData,实际上这个单例是软引用的,

解释一下各种引用:

⑴强引用(StrongReference)
强引用是使用最普遍的引用。如果一个对象具有强引用,那垃圾回收器绝不会回收它。当内存空间不足,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足的问题。

⑵软引用(SoftReference)
如果一个对象只具有软引用,则内存空间足够,垃圾回收器就不会回收它;如果内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就可以被程序使用。软引用可用来实现内存敏感的高速缓存。
软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所引用的对象被垃圾回收器回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列中。

⑶弱引用(WeakReference)
弱引用与软引用的区别在于:只具有弱引用的对象拥有更短暂的生命周期。在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。
弱引用可以和一个引用队列(ReferenceQueue)联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。

⑷虚引用(PhantomReference)
“虚引用”顾名思义,就是形同虚设,与其他几种引用都不同,虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。
虚引用主要用来跟踪对象被垃圾回收器回收的活动。虚引用与软引用和弱引用的一个区别在于:虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之 关联的引用队列中。

具体获取数据库中信息的代码为以下:(DBQueryRequest是封装的数据库查询类,keys是一个含有一个string类型标记值的数组)

  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();

为什么这么写呢?

其实在类图和流程图中个各类的分工已经很明确,完全符合MVVM架构结合liveData的实现方式,

  • View(activity、fragment)
    负责了数据在数据出口输出时响应回调,进行数据刷新,
  • ViewModel
    负责实例化各种存储数据的容器,即liveData,包括各个liveData之间的数据传值和包装逻辑
  • Model用自定义LiveData代替
    实现数据请求等操作

其实主要是想利用livedata的生命周期,从而实现数据的自动传递。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容