RecycleView的有效埋點(diǎn)問(wèn)題

問(wèn)題

PM需要獲取當(dāng)前條目的有效曝光給大數(shù)據(jù)分析推廣適用,因此需要獲取recycleView的有效曝光的埋點(diǎn)數(shù)據(jù);

  • 要求
    1. RecycleView中復(fù)用條目不用重復(fù)埋點(diǎn),除非下拉刷新數(shù)據(jù);
    2. 待確定:條目UI顯示超過(guò)50%方可埋點(diǎn),否則不埋點(diǎn);

分析

由于RecycleView的四級(jí)緩存機(jī)制,當(dāng)我們?cè)趏nBinding中綁定數(shù)據(jù)時(shí)埋點(diǎn)會(huì)增加二級(jí)緩存的埋點(diǎn),導(dǎo)致獲取有效曝光不準(zhǔn)確問(wèn)題?如何解決該問(wèn)題:兩種方式

View繪制流程
  • 平臺(tái)測(cè)目前在用重寫onAttachedToWindow()和onDetachedFromWindow()這兩個(gè)方法在RecyclerView內(nèi)部會(huì)在View移動(dòng)出可視區(qū)域的時(shí)候被觸發(fā);
    1. 當(dāng) Adapter 創(chuàng)建的 View 在被滑動(dòng)進(jìn)屏幕的時(shí)onViewAttachedToWindow() 會(huì)直接回調(diào),反之,在列表項(xiàng) View 被窗口分離(即滑動(dòng)離開(kāi)了當(dāng)前窗口界面的)的時(shí)onViewDetachedToWindow() 會(huì)立馬被調(diào)用。
  1. 根據(jù)以上特性,在adapter中重寫onViewAttachedToWindow(RecycleView.ViewHolder)可以獲取當(dāng)前列表剛剛滑進(jìn)屏幕的條目布局信息,那么埋點(diǎn)的數(shù)據(jù)如何綁定?
    • 重寫viewHolder通過(guò)tag保存和讀取,平臺(tái)已經(jīng)封裝ViewHolder,需要修改每個(gè)Delegate中的viewHolder繼承該類
    public class ViewHolder extends androidx.recyclerview.widget.RecyclerView.ViewHolder {
     private SparseArray<View> mViews = new SparseArray();
     private SparseArray<Object> mKeyedTags;
    
     public ViewHolder(View itemView) {
         super(itemView);
     }
    
     public void setTag(int key, Object tag) {
         if (key >>> 24 < 2) {
             throw new IllegalArgumentException("The key must be an application-specific resource id.");
         } else {
             if (this.mKeyedTags == null) {
                 this.mKeyedTags = new SparseArray(2);
             }
    
             this.mKeyedTags.put(key, tag);
         }
     }
    
     public Object getTag(int key) {
         return this.mKeyedTags != null ? this.mKeyedTags.get(key) : null;
     }
    
    1. adapter通過(guò)Delegate添加每個(gè)條目布局和數(shù)據(jù),然后在Delegate的 onBindViewHolder中設(shè)置tag屬性,并將埋點(diǎn)所需的條目數(shù)據(jù)添加進(jìn)去
public class PtClientAdapter extends JobPtAbsDelegationAdapter {
    public PtClientAdapter(Activity activity, List<PtCateListBean.PtBaseListBean> items, OnOptCallBack onOptCallBack,
                           OnItemClickCallback onItemClickCallback,ActionUniteInterface callBack) {

        this.delegatesManager.addDelegate(new PtClientNormalDelegate(activity , mCallBack)); //普通兼職職位
        this.delegatesManager.addDelegate(new PtListBannersDelegate(activity , mCallBack));//輪播圖
        this.delegatesManager.addDelegate(new PtOnlineTaskDelegate(activity, onItemClickCallback , mCallBack));//線上任務(wù)
        this.delegatesManager.addDelegate(new PtHotCateDelegate(activity, onFilterCallback , mCallBack)); //你可能在找
        this.delegatesManager.addDelegate(new PtEncourageVideoDelegate(activity , mCallBack));//激勵(lì)視頻
        this.delegatesManager.addDelegate(new PtResumeDelegate(activity , mCallBack)); //簡(jiǎn)歷引導(dǎo)
        this.delegatesManager.addDelegate(new PtCustomDelegate(activity , mCallBack)); //會(huì)員定制
        this.delegatesManager.addDelegate(new PtOperatingItemDelegate(activity , mCallBack)); //猜你喜歡
    }
}

//在Delegate中設(shè)置tag
public class PtClientNormalDelegate extends AdapterDelegate{
    @Override
    protected void onBindViewHolder(@NonNull List<PtCateListBean.PtBaseListBean> items, final int position, @NonNull RecyclerView.ViewHolder holder, @NonNull List<Object> payloads) {
          
        final PtCateListBean.PositionNormal positionNormalBean = (PtCateListBean.PositionNormal) items.get(position);
        final NormalViewHolder viewHolder = (NormalViewHolder) holder;
        //設(shè)置tag,并把當(dāng)前條目信息加入緩存
         viewHolder.setTag(R.id.id_tag_detail_bean, positionNormalBean);
    }
}
  1. 由于每個(gè)Delegate對(duì)應(yīng)的javaBean對(duì)象類都是不同,直接寫到adapter中會(huì)導(dǎo)致無(wú)法很輕松理解,平臺(tái)已經(jīng)封裝過(guò)了,在Adapter中通過(guò)DeleGateManager將onViewAttachedToWindow()分發(fā)給每一個(gè)Delegate類,因此可以直接重寫Delegate的onViewAttachedToWindow(ViewHolder holder)
 @Override
 protected void onViewAttachedToWindow(@NonNull RecyclerView.ViewHolder holder) {
     super.onViewAttachedToWindow(holder);
     ViewHolder viewHolder = (ViewHolder) holder;

     Object tag = viewHolder.getTag(R.id.id_tag_detail_bean);

     if (tag instanceof  PtCateListBean.PositionNormal) {
         PtCateListBean.PositionNormal positionNormalBean = (PtCateListBean.PositionNormal) tag;
         int adapterPosition = viewHolder.getAdapterPosition();
         Log.e("shiq" , "當(dāng)前被顯示了 - onViewAttachedToWindow : " + positionNormalBean.title  + "   " + positionNormalBean + " ---- 列表中的位置為: " + adapterPosition);
     }
 }
  • 以上對(duì)于每個(gè)Delegate都在自己類中添加有效埋點(diǎn)數(shù)據(jù)。便于后期維護(hù),但有一個(gè)問(wèn)題,PM要求相同的埋點(diǎn)滑動(dòng)時(shí)只埋一次。onViewAttachedToWindow會(huì)每次顯示均會(huì)調(diào)用一次。如何解決呢?
    1. 在javaBean中設(shè)置boolean值記錄當(dāng)前是否首次顯示被埋點(diǎn)過(guò)了,如果埋點(diǎn)標(biāo)記為true,后續(xù)顯示均不會(huì)埋點(diǎn)了:優(yōu)點(diǎn)簡(jiǎn)單,缺點(diǎn)如果javaBean是三方的不易修改;
    2. 在adapter中創(chuàng)建集合記錄已經(jīng)被標(biāo)記埋點(diǎn)過(guò)的javaBean數(shù)據(jù),如何區(qū)分javaBean唯一性,可以通過(guò)hashCode + position標(biāo)記,如果首次顯示埋點(diǎn)后添加記錄,再次顯示后過(guò)濾掉即可: 優(yōu)點(diǎn):不修改原有數(shù)據(jù),缺點(diǎn):每次都需判斷是否在集合中,性能有所影響;
  • 不足之處: onViewAttachedToWindow無(wú)法區(qū)分當(dāng)前UI是否被顯示超過(guò)50%;

通過(guò)RecycleView的滑動(dòng)監(jiān)聽(tīng)

  • 通過(guò)監(jiān)聽(tīng)RecycleView的滑動(dòng)事件,獲取當(dāng)前屏幕顯示的條目信息,根據(jù)條件刪選即可!
  1. 重寫RecycleView的onScrollStateChanged,onScrolled方法
 @Override
    public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
        switch (newState) {
            case RecyclerView.SCROLL_STATE_IDLE:
//            case RecyclerView.SCROLL_STATE_DRAGGING:
//            case RecyclerView.SCROLL_STATE_SETTLING:
                findScreenVisibleViewsAndNotify();
                break;
        }
    }

    @Override
    public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) {
        super.onScrolled(recyclerView, dx, dy);
        if (dx == 0 && dy == 0) { //如果當(dāng)前是首次進(jìn)入時(shí)設(shè)置
            findScreenVisibleViewsAndNotify();
        }
    }
  1. RecycleVeiw的LinearLayoutManager布局獲取當(dāng)前屏幕顯示的首位和末位的條目,不足: 結(jié)果并不準(zhǔn)確,findLastVisibleItemPosition大于當(dāng)前顯示位置;
 range[0] = manager.findFirstVisibleItemPosition();
 range[1] = manager.findLastVisibleItemPosition();
  1. 對(duì)于上述中的不足之處,我們應(yīng)該如何優(yōu)化使之適合我們的要求,這里用到了view.getGlobalVisibleRect()獲取的是view可見(jiàn)區(qū)域相對(duì)與屏幕來(lái)說(shuō)的坐標(biāo)位置;


    image
 Rect rect = new Rect();
 boolean cover = view.getGlobalVisibleRect(rect);

  //item邏輯上可見(jiàn):可見(jiàn)且可見(jiàn)高度(寬度)>view高度(寬度)50%才行
  boolean visibleHeightEnough = orientation == OrientationHelper.VERTICAL && rect.height() > view.getMeasuredHeight() / 2;
 boolean visibleWidthEnough = orientation == OrientationHelper.HORIZONTAL && rect.width() > view.getMeasuredWidth() / 2;
 boolean isItemViewVisibleInLogic = visibleHeightEnough || visibleWidthEnough;
 if (cover  && isItemViewVisibleInLogic) {
    //去重,可埋點(diǎn)的數(shù)據(jù)
}
  1. 我們已經(jīng)獲取到了當(dāng)前坐標(biāo)position是否被顯示且滿足條件,對(duì)于去重,依然采用View繪制中兩種方式,這里使用第二種,通過(guò)集合保存已被埋點(diǎn)數(shù)據(jù),定義統(tǒng)一接口給adapter適配用于數(shù)據(jù)獲?。?/li>
//數(shù)據(jù)區(qū)分接口
public interface IRecyclerViewAdapter {
    /**
     * 根據(jù)position獲取item的數(shù)據(jù)
     */
    Object getCurrentItemData(int position);

    int getCurrentSize();

}
// 獲取到需展示數(shù)據(jù)接口
public interface OnRecycleExposureListener {

    /**
     * 當(dāng)前被展示的數(shù)據(jù)集合
     * @param exposureBeans
     */
    void onExposure(List<ExposureBean> exposureBeans);
}

Object itemData = null;
if (cover  && isItemViewVisibleInLogic) {
    RecyclerView.Adapter adapter = mRecyclerView.getAdapter();
    if (adapter != null && adapter instanceof IRecyclerViewAdapter){
         int currentSize = ((IRecyclerViewAdapter) adapter).getCurrentSize();
         if (currentSize > position){
             itemData = ((IRecyclerViewAdapter) adapter).getCurrentItemData(position);
          }
    }
}

if (itemData == null)  return;//如果不存在數(shù)據(jù),跳過(guò)本次循環(huán)

if (mManager.addResource(mRule.createItemID(itemData, position))) {
      mAllShowList.add(new ExposureBean(itemData, view, position));
 }
  • 提供通用的去重規(guī)則接口,便于后續(xù)擴(kuò)展,這里使用規(guī)則為javaBean的hanshCode + position
  • 數(shù)據(jù)管理集合,由于每次均需要查詢是否在其中,這里為了效率mManager推薦使用hashSet,盡量避免使用ArrayList,當(dāng)列表數(shù)據(jù)過(guò)大時(shí)會(huì)影響效率!
  • 獲取到的數(shù)據(jù)保存在mAllShowList集合中,通過(guò)接口回掉或者動(dòng)態(tài)代理(如果不太清楚adapter類型,在view層通過(guò) instanceof OnRecycleExposureListener)
  1. 在adapter中實(shí)現(xiàn)接口,分發(fā)給每個(gè)Delegate去埋點(diǎn),也可以通過(guò)view.setTag和getTag使用獲??;
public void onExposure(List<ExposureBean> exposureBeans) {

   if (exposureBeans != null && !exposureBeans.isEmpty()) {
        for (ExposureBean bean : exposureBeans) {
          //根據(jù)當(dāng)前位置獲取設(shè)置AdapterDelegate
           int itemViewType = this.delegatesManager.getItemViewType(items, bean.position);
           AdapterDelegate delegateForViewType = this.delegatesManager.getDelegateForViewType(itemViewType);
           if (bean.itemData != null && delegateForViewType != null)
             delegateForViewType.exPostActionItem(bean.itemData, bean.position);
        }
    }
}
  • 總結(jié): RecycleView的adapter實(shí)現(xiàn)IRecyclerViewAdapter提供去重?cái)?shù)據(jù),OnRecycleExposureListener返回需要埋點(diǎn)集合,每個(gè)Delegate重寫exPostActionItem方法去添加有效曝光的埋點(diǎn)即可!
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

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

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