問(wèn)題
PM需要獲取當(dāng)前條目的有效曝光給大數(shù)據(jù)分析推廣適用,因此需要獲取recycleView的有效曝光的埋點(diǎn)數(shù)據(jù);
- 要求
- RecycleView中復(fù)用條目不用重復(fù)埋點(diǎn),除非下拉刷新數(shù)據(jù);
- 待確定:條目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ā);
- 當(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)用。
- 根據(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; }- 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);
}
}
- 由于每個(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)用一次。如何解決呢?
- 在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是三方的不易修改;
- 在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ù)條件刪選即可!
- 重寫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();
}
}
- RecycleVeiw的LinearLayoutManager布局獲取當(dāng)前屏幕顯示的首位和末位的條目,不足: 結(jié)果并不準(zhǔn)確,findLastVisibleItemPosition大于當(dāng)前顯示位置;
range[0] = manager.findFirstVisibleItemPosition();
range[1] = manager.findLastVisibleItemPosition();
-
對(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ù)
}
- 我們已經(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)
- 在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)即可!
