網站首頁 編程語言 正文
本文實例為大家分享了Android RecyclerView曝光采集的具體代碼,供大家參考,具體內容如下
一、背景
近期pm提出需要統計首頁商品的曝光亮,由于我們的首頁是用的recylerview實現的,這里就來講下如何使用監聽recylerview的滾動事件來實現子view的曝光量統計,我們這里說的view都是列表中的子item條目(子view)
二、監聽recylerview的滾動事件OnScrollListener
onScrollStateChanged:監聽滾動狀態
onScrolled:監聽滾動
我們接下來的統計工作,就是拿這兩個方法做文章。
//檢測recylerview的滾動事件 recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() { ? ? ? ?@Override ? ? ? ?public void onScrollStateChanged(RecyclerView recyclerView, int newState) { ? ? ? ? ? ?/* ? ? ? ? ? ?我這里通過的是停止滾動后屏幕上可見view。如果滾動過程中的可見view也要統計,你可以根據newState去做區分 ? ? ? ? ? ?SCROLL_STATE_IDLE:停止滾動 ? ? ? ? ? ?SCROLL_STATE_DRAGGING: 用戶慢慢拖動 ? ? ? ? ? ?SCROLL_STATE_SETTLING:慣性滾動 ? ? ? ? ? ?*/ ? ? ? ? ? ?if (newState == RecyclerView.SCROLL_STATE_IDLE) { ? ? ? ? ? ? ? ?..... ? ? ? ? ? ?} ? ? ? ?} ? ? ? ?@Override ? ? ? ?public void onScrolled(RecyclerView recyclerView, int dx, int dy) { ? ? ? ? ? ?super.onScrolled(recyclerView, dx, dy); ? ? ? ? ? ........ ? ? ? ?} ? ?});
首先再次明確下,我們要統計的是用戶停止滑動時,顯示在屏幕的上控件。所以我們要監測到onScrollStateChanged 方法中
newState == RecyclerView.SCROLL_STATE_IDLE 時,也就是用戶停止滾動。然后在這里做文章
三、獲取屏幕內可見條目的起始位置
這里的起始位置就是指我們屏幕當中最上面和最下面條目的位置。比如下圖的0就是最上面的可見條目,3就是最下面的可見條目。我們次數的曝光view就是0,1,2,3 這個時候這四個條目顯示在屏幕中。我們這時就要對這4個view的曝光量進行加1
那么接下來的重點就是要去獲取屏幕內可見條目的起始位置。獲取到起始位置后,當前屏幕里的可見條目就都能拿到了。
而recylerview的manager正好給我們提供的有對應的方法。
findFirstVisibleItemPosition()和findLastVisibleItemPosition() 看字面意思就能知道這時干嘛用的。
但是我們的manager不止LinearLayoutManager一種,所以我們要做下區分
//這里我們用一個數組來記錄起始位置 int[] range = new int[2]; RecyclerView.LayoutManager manager = reView.getLayoutManager(); if (manager instanceof LinearLayoutManager) { ? ? range = findRangeLinear((LinearLayoutManager) manager); } else if (manager instanceof GridLayoutManager) { ? ? range = findRangeGrid((GridLayoutManager) manager); } else if (manager instanceof StaggeredGridLayoutManager) { ? ? range = findRangeStaggeredGrid((StaggeredGridLayoutManager) manager); }
LinearLayoutManager和GridLayoutManager獲取起始位置方法如下
private int[] findRangeLinear(LinearLayoutManager manager) { ? ? int[] range = new int[2]; ? ? range[0] = manager.findFirstVisibleItemPosition(); ? ? range[1] = manager.findLastVisibleItemPosition(); ? ? return range; } private int[] findRangeGrid(GridLayoutManager manager) { ? ? int[] range = new int[2]; ? ? range[0] = manager.findFirstVisibleItemPosition(); ? ? range[1] = manager.findLastVisibleItemPosition(); ? ? return range; }
StaggeredGridLayoutManager獲取起始位置有點復雜,如下
private int[] findRangeStaggeredGrid(StaggeredGridLayoutManager manager) { ? ? int[] startPos = new int[manager.getSpanCount()]; ? ? int[] endPos = new int[manager.getSpanCount()]; ? ? manager.findFirstVisibleItemPositions(startPos); ? ? manager.findLastVisibleItemPositions(endPos); ? ? int[] range = findRange(startPos, endPos); ? ? return range; } private int[] findRange(int[] startPos, int[] endPos) { ? ?int start = startPos[0]; ? ? int end = endPos[0]; ? ? for (int i = 1; i < startPos.length; i++) { ? ? ? ? if (start > startPos[i]) { ? ? ? ? ? ? start = startPos[i]; ? ? ? ? } ? ? } ? ? for (int i = 1; i < endPos.length; i++) { ? ? ? ? if (end < endPos[i]) { ? ? ? ? ? ? end = endPos[i]; ? ? ? ? } ? ? } ? ? int[] res = new int[]{start, end}; ? ? return res; }
四、獲取到起始位置以后,我們就根據位置獲取到view及view中的數據
上面第三步拿到屏幕內可見條目的起始位置以后,我們就用一個for循環,獲取當前屏幕內可見的所有子view
for (int i = range[0]; i <= range[1]; i++) { ?View view = manager.findViewByPosition(i); ? recordViewCount(view); }
recordViewCount是我自己寫的用于獲取子view內綁定數據的方法
//獲取view綁定的數據 private void recordViewCount(View view) { ? ? if (view == null || view.getVisibility() != View.VISIBLE || ? ? ? ? ? ? !view.isShown() || !view.getGlobalVisibleRect(new Rect())) { ? ? ? ? return; ? ? } ? ? int top = view.getTop(); ? ? int halfHeight = view.getHeight() / 2; ? ? int screenHeight = UiUtils.getScreenHeight((Activity) view.getContext()); ? ? int statusBarHeight = UiUtils.getStatusBarHeight(view.getContext()); ? ? if (top < 0 && Math.abs(top) > halfHeight) { ? ? ? ? return; ? ? } ? ? if (top > screenHeight - halfHeight - statusBarHeight) { ? ? ? ? return; ? ? } ? ? //這里獲取的是我們view綁定的數據,相應的你要去在你的view里setTag,只有set了,才能get ? ? ItemData tag = (ItemData) view.getTag(); ? ? String key = tag.toString(); ? ? if (TextUtils.isEmpty(key)) { ? ? ? ? return; ? ? } ? ? hashMap.put(key, !hashMap.containsKey(key) ? 1 : (hashMap.get(key) + 1)); ? ? Log.i("qcl0402", key + "----出現次數:" + hashMap.get(key)); }
這里有幾點需要注意
這這里起始位置的view顯示區域如果不超過50%,就不算這個view可見,進而也就不統計曝光。
我們通過view.getTag();獲取view里的數據,必須在此之前setTag()數據,我這里setTag是在viewholder中把數據set進去的
到這里我們就實現了recylerview列表中view控件曝光量的統計了。下面貼出來完整的代碼給大家
package com.example.qcl.demo.xuexi.baoguang; import android.app.Activity; import android.graphics.Rect; import android.support.v7.widget.GridLayoutManager; import android.support.v7.widget.LinearLayoutManager; import android.support.v7.widget.RecyclerView; import android.support.v7.widget.StaggeredGridLayoutManager; import android.text.TextUtils; import android.util.Log; import android.view.View; import com.example.qcl.demo.utils.UiUtils; import java.util.concurrent.ConcurrentHashMap; /** ?* 2019/4/2 13:31 ?* author: qcl ?* desc: 安卓曝光量統計工具類 ?* wechat:2501902696 ?*/ public class ViewShowCountUtils { ? ? //剛進入列表時統計當前屏幕可見views ? ? private boolean isFirstVisible = true; ? ? //用于統計曝光量的map ? ? private ConcurrentHashMap<String, Integer> hashMap = new ConcurrentHashMap<String, Integer>(); ? ? /* ? ? ?* 統計RecyclerView里當前屏幕可見子view的曝光量 ? ? ?* ? ? ?* */ ? ? void recordViewShowCount(RecyclerView recyclerView) { ? ? ? ? hashMap.clear(); ? ? ? ? if (recyclerView == null || recyclerView.getVisibility() != View.VISIBLE) { ? ? ? ? ? ? return; ? ? ? ? } ? ? ? ? //檢測recylerview的滾動事件 ? ? ? ? recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() { ? ? ? ? ? ? @Override ? ? ? ? ? ? public void onScrollStateChanged(RecyclerView recyclerView, int newState) { ? ? ? ? ? ? ? ? /* ? ? ? ? ? ? ? ? 我這里通過的是停止滾動后屏幕上可見view。如果滾動過程中的可見view也要統計,你可以根據newState去做區分 ? ? ? ? ? ? ? ? SCROLL_STATE_IDLE:停止滾動 ? ? ? ? ? ? ? ? SCROLL_STATE_DRAGGING: 用戶慢慢拖動 ? ? ? ? ? ? ? ? SCROLL_STATE_SETTLING:慣性滾動 ? ? ? ? ? ? ? ? */ ? ? ? ? ? ? ? ? if (newState == RecyclerView.SCROLL_STATE_IDLE) { ? ? ? ? ? ? ? ? ? ? getVisibleViews(recyclerView); ? ? ? ? ? ? ? ? } ? ? ? ? ? ? } ? ? ? ? ? ? @Override ? ? ? ? ? ? public void onScrolled(RecyclerView recyclerView, int dx, int dy) { ? ? ? ? ? ? ? ? super.onScrolled(recyclerView, dx, dy); ? ? ? ? ? ? ? ? //剛進入列表時統計當前屏幕可見views ? ? ? ? ? ? ? ? if (isFirstVisible) { ? ? ? ? ? ? ? ? ? ? getVisibleViews(recyclerView); ? ? ? ? ? ? ? ? ? ? isFirstVisible = false; ? ? ? ? ? ? ? ? } ? ? ? ? ? ? } ? ? ? ? }); ? ? } ? ? /* ? ? ?* 獲取當前屏幕上可見的view ? ? ?* */ ? ? private void getVisibleViews(RecyclerView reView) { ? ? ? ? if (reView == null || reView.getVisibility() != View.VISIBLE || ? ? ? ? ? ? ? ? !reView.isShown() || !reView.getGlobalVisibleRect(new Rect())) { ? ? ? ? ? ? return; ? ? ? ? } ? ? ? ? //保險起見,為了不讓統計影響正常業務,這里做下try-catch ? ? ? ? try { ? ? ? ? ? ? int[] range = new int[2]; ? ? ? ? ? ? RecyclerView.LayoutManager manager = reView.getLayoutManager(); ? ? ? ? ? ? if (manager instanceof LinearLayoutManager) { ? ? ? ? ? ? ? ? range = findRangeLinear((LinearLayoutManager) manager); ? ? ? ? ? ? } else if (manager instanceof GridLayoutManager) { ? ? ? ? ? ? ? ? range = findRangeGrid((GridLayoutManager) manager); ? ? ? ? ? ? } else if (manager instanceof StaggeredGridLayoutManager) { ? ? ? ? ? ? ? ? range = findRangeStaggeredGrid((StaggeredGridLayoutManager) manager); ? ? ? ? ? ? } ? ? ? ? ? ? if (range == null || range.length < 2) { ? ? ? ? ? ? ? ? return; ? ? ? ? ? ? } ? ? ? ? ? ? Log.i("qcl0402", "屏幕內可見條目的起始位置:" + range[0] + "---" + range[1]); ? ? ? ? ? ? for (int i = range[0]; i <= range[1]; i++) { ? ? ? ? ? ? ? ? View view = manager.findViewByPosition(i); ? ? ? ? ? ? ? ? recordViewCount(view); ? ? ? ? ? ? } ? ? ? ? } catch (Exception e) { ? ? ? ? ? ? e.printStackTrace(); ? ? ? ? } ? ? } ? ? //獲取view綁定的數據 ? ? private void recordViewCount(View view) { ? ? ? ? if (view == null || view.getVisibility() != View.VISIBLE || ? ? ? ? ? ? ? ? !view.isShown() || !view.getGlobalVisibleRect(new Rect())) { ? ? ? ? ? ? return; ? ? ? ? } ? ? ? ? int top = view.getTop(); ? ? ? ? int halfHeight = view.getHeight() / 2; ? ? ? ? int screenHeight = UiUtils.getScreenHeight((Activity) view.getContext()); ? ? ? ? int statusBarHeight = UiUtils.getStatusBarHeight(view.getContext()); ? ? ? ? if (top < 0 && Math.abs(top) > halfHeight) { ? ? ? ? ? ? return; ? ? ? ? } ? ? ? ? if (top > screenHeight - halfHeight - statusBarHeight) { ? ? ? ? ? ? return; ? ? ? ? } ? ? ? ? //這里獲取的是我們view綁定的數據,相應的你要去在你的view里setTag,只有set了,才能get ? ? ? ? ItemData tag = (ItemData) view.getTag(); ? ? ? ? String key = tag.toString(); ? ? ? ? if (TextUtils.isEmpty(key)) { ? ? ? ? ? ? return; ? ? ? ? } ? ? ? ? hashMap.put(key, !hashMap.containsKey(key) ? 1 : (hashMap.get(key) + 1)); ? ? ? ? Log.i("qcl0402", key + "----出現次數:" + hashMap.get(key)); ? ? } ? ? private int[] findRangeLinear(LinearLayoutManager manager) { ? ? ? ? int[] range = new int[2]; ? ? ? ? range[0] = manager.findFirstVisibleItemPosition(); ? ? ? ? range[1] = manager.findLastVisibleItemPosition(); ? ? ? ? return range; ? ? } ? ? private int[] findRangeGrid(GridLayoutManager manager) { ? ? ? ? int[] range = new int[2]; ? ? ? ? range[0] = manager.findFirstVisibleItemPosition(); ? ? ? ? range[1] = manager.findLastVisibleItemPosition(); ? ? ? ? return range; ? ? } ? ? private int[] findRangeStaggeredGrid(StaggeredGridLayoutManager manager) { ? ? ? ? int[] startPos = new int[manager.getSpanCount()]; ? ? ? ? int[] endPos = new int[manager.getSpanCount()]; ? ? ? ? manager.findFirstVisibleItemPositions(startPos); ? ? ? ? manager.findLastVisibleItemPositions(endPos); ? ? ? ? int[] range = findRange(startPos, endPos); ? ? ? ? return range; ? ? } ? ? private int[] findRange(int[] startPos, int[] endPos) { ? ? ? ? int start = startPos[0]; ? ? ? ? int end = endPos[0]; ? ? ? ? for (int i = 1; i < startPos.length; i++) { ? ? ? ? ? ? if (start > startPos[i]) { ? ? ? ? ? ? ? ? start = startPos[i]; ? ? ? ? ? ? } ? ? ? ? } ? ? ? ? for (int i = 1; i < endPos.length; i++) { ? ? ? ? ? ? if (end < endPos[i]) { ? ? ? ? ? ? ? ? end = endPos[i]; ? ? ? ? ? ? } ? ? ? ? } ? ? ? ? int[] res = new int[]{start, end}; ? ? ? ? return res; ? ? } }
原文鏈接:https://blog.csdn.net/weixin_42600398/article/details/119796478
相關推薦
- 2022-08-04 PyTorch中torch.manual_seed()的用法實例詳解_python
- 2022-06-09 基于Python實現對比Exce的工具_python
- 2024-03-08 jdk版本不兼容 Error creating bean with name ‘springSecu
- 2022-09-26 Redis刪除策略的三種方法及逐出算法_Redis
- 2023-02-03 Linux設置每晚定時備份Oracle數據表的操作命令_linux shell
- 2024-02-26 JqGrid獲得所有選中行數據ID數組,獲取所有行的ID數組
- 2022-11-11 python?使用第三方庫requests-toolbelt?上傳文件流的示例_python
- 2021-12-04 C#獲取Windows10屏幕縮放比例的操作方法_C#教程
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支