Android

RecycleView的有效埋点问题

2021-08-25  本文已影响0人  唯爱_0834

问题

PM需要获取当前条目的有效曝光给大数据分析推广适用,因此需要获取recycleView的有效曝光的埋点数据;

分析

由于RecycleView的四级缓存机制,当我们在onBinding中绑定数据时埋点会增加二级缓存的埋点,导致获取有效曝光不准确问题?如何解决该问题:两种方式

View绘制流程
  1. 根据以上特性,在adapter中重写onViewAttachedToWindow(RecycleView.ViewHolder)可以获取当前列表刚刚滑进屏幕的条目布局信息,那么埋点的数据如何绑定?
    • 重写viewHolder通过tag保存和读取,平台已经封装ViewHolder,需要修改每个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通过Delegate添加每个条目布局和数据,然后在Delegate的 onBindViewHolder中设置tag属性,并将埋点所需的条目数据添加进去
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));//线上任务
        this.delegatesManager.addDelegate(new PtHotCateDelegate(activity, onFilterCallback , mCallBack)); //你可能在找
        this.delegatesManager.addDelegate(new PtEncourageVideoDelegate(activity , mCallBack));//激励视频
        this.delegatesManager.addDelegate(new PtResumeDelegate(activity , mCallBack)); //简历引导
        this.delegatesManager.addDelegate(new PtCustomDelegate(activity , mCallBack)); //会员定制
        this.delegatesManager.addDelegate(new PtOperatingItemDelegate(activity , mCallBack)); //猜你喜欢
    }
}

//在Delegate中设置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;
        //设置tag,并把当前条目信息加入缓存
         viewHolder.setTag(R.id.id_tag_detail_bean, positionNormalBean);
    }
}
  1. 由于每个Delegate对应的javaBean对象类都是不同,直接写到adapter中会导致无法很轻松理解,平台已经封装过了,在Adapter中通过DeleGateManager将onViewAttachedToWindow()分发给每一个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" , "当前被显示了 - onViewAttachedToWindow : " + positionNormalBean.title  + "   " + positionNormalBean + " ---- 列表中的位置为: " + adapterPosition);
     }
 }

通过RecycleView的滑动监听

  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) { //如果当前是首次进入时设置
            findScreenVisibleViewsAndNotify();
        }
    }
  1. RecycleVeiw的LinearLayoutManager布局获取当前屏幕显示的首位和末位的条目,不足: 结果并不准确,findLastVisibleItemPosition大于当前显示位置;
 range[0] = manager.findFirstVisibleItemPosition();
 range[1] = manager.findLastVisibleItemPosition();
  1. 对于上述中的不足之处,我们应该如何优化使之适合我们的要求,这里用到了view.getGlobalVisibleRect()获取的是view可见区域相对与屏幕来说的坐标位置;


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

  //item逻辑上可见:可见且可见高度(宽度)>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) {
    //去重,可埋点的数据
}
  1. 我们已经获取到了当前坐标position是否被显示且满足条件,对于去重,依然采用View绘制中两种方式,这里使用第二种,通过集合保存已被埋点数据,定义统一接口给adapter适配用于数据获取;
//数据区分接口
public interface IRecyclerViewAdapter {
    /**
     * 根据position获取item的数据
     */
    Object getCurrentItemData(int position);

    int getCurrentSize();

}
// 获取到需展示数据接口
public interface OnRecycleExposureListener {

    /**
     * 当前被展示的数据集合
     * @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;//如果不存在数据,跳过本次循环

if (mManager.addResource(mRule.createItemID(itemData, position))) {
      mAllShowList.add(new ExposureBean(itemData, view, position));
 }
  1. 在adapter中实现接口,分发给每个Delegate去埋点,也可以通过view.setTag和getTag使用获取;
public void onExposure(List<ExposureBean> exposureBeans) {

   if (exposureBeans != null && !exposureBeans.isEmpty()) {
        for (ExposureBean bean : exposureBeans) {
          //根据当前位置获取设置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);
        }
    }
}
上一篇下一篇

猜你喜欢

热点阅读