揭秘!如何用Flutter设计一个100%准确的埋点框架?
阿里妹导读:用户行为埋点是用来记录用户在操作时的一系列行为,也是业务做判断的核心数据依据,如果缺失或者不准确将会给业务带来不可恢复的损失。闲鱼将业务代码从Native迁移到Flutter上过程中,发现原先Native体系上的埋点方案无法应用在Flutter体系之上。而如果只把业务功能迁移过来就上线,是极其不负责任的。因此,经过不断探索,闲鱼技术团队沉淀了一套Flutter上的高准确率的用户行为埋点方案,今天由工程师兰昊来和大家分享一下。
用户行为埋点定位
先来讲讲在我们这里是如何定义用户行为埋点的。在如下用户时间轴上,用户进入A页面后,看到了按钮X,然后点击了这个按钮,随即打开了新的页面B。
这个时间轴上有如下5个埋点事件发生:
- 进入A页面。A页面首帧渲染完毕,并获得了焦点。
- 曝光坑位X。按钮X处于手机屏幕内,且停留一段时间,让用户可见可触摸。
- 点击坑位X。用户对按钮X的内容很感兴趣,于是点击了它。按钮X响应点击,然后需要打开一个新页面。
- 离开A页面。A页面失去焦点。
- 进入B页面。B页面首帧渲染完毕,并获得焦点。
在这里,打埋点最重要的是时机,即在什么时机下的事件中触发什么埋点,下面来看看闲鱼在Flutter上的实现方案。
实现方案
进入/离开页面
在Native原生开发中,Android端是监听Activity的onResume和onPause事件来做为页面的进入和离开事件,同理iOS端是监听UIViewController的viewWillAppear和viewDidDisappear事件来做为页面的进入和离开事件。同时整个页面栈是由Android和iOS操作系统来维护。
在Flutter中,Android和iOS端分别是用FlutterActivity和FlutterViewController来做为容器承载Flutter的页面,通过这个容器可以在一个Native的页面内来进行Flutter页面的切换,即Flutter自己维护了一个Flutter页面的页面栈。这样,原来我们最熟悉的那套在Native原生上的方案在Flutter上无法直接运作起来。
针对这个问题,可能很多人会想到去注册监听Flutter的NavigatorObserver,这样就知道Flutter页面的进栈(push)和出栈(pop)事件。但是这会有两个问题:
- 假设A、B两个页面先后进栈(A enter -> A leave -> B enter)。然后B页面返回退出(B leave),此时A页面重新可见,但是此时是收不到A页面push(A enter)的事件。
- 假设在A页面弹出一个Dialog或者BottomSheet,而这两类也会走push操作,但实际上A页面并未离开。
好在Flutter的页面栈不像Android Native的页面栈那么复杂,所以针对第一个问题,我们可以维护一个和页面栈匹配的索引列表。当收到A页面的push事件时,往队列里塞入A的索引。当收到B页面的push事件时,检测列表内是否有页面,如有,则对列表最后一个页面执行离开页面事件,再对B页面执行进入页面事件,接着往队列里塞B的索引。当收到B页面的pop事件时,先对B页面执行离开页面事件记录,再对队列里存在的最后一个索引对应的页面(假设为A)进行判断是否在栈顶(ModalRoute.of(context).isCurrent),如果是,则对A页面执行进入页面事件。
针对第二个问题,Route类内有个成员变量overlayEntries,可以获取当前Route对应的所有图层OverlayEntry,在OverlayEntry对象中有个成员变量opaque可以判断当前这个图层是否全屏覆盖,从而可以排除Dialog和BottomSheet这种类型。再结合问题1,还需要在上述方案中加上对push进来的新页面来做判断是否为一个有效页面。如果是有效页面,才对索引列表中前一个页面做离开页面事件,且将有效页面加到索引列表中。如果不是有效页面,则不操作索引列表。
以上并不是闲鱼的方案,只是笔者给出的一个建议。因为闲鱼APP在一开始落地Flutter框架时,就没有使用Flutter原生的页面栈管理方案,而是采用了Native+Flutter混合开发的方案,因此接下来也是基于此来阐述闲鱼的方案。
闲鱼的方案如下(以Android为例,iOS同理):
注:首次打开指的是基于混合栈新打开一个页面,非首次打开指的是通过回退页面的方式,在后台的页面再次到前台可见。
看到这个方案可能会有人问,为什么这么绕,为什么不全部交给Native侧去直接管理呢?交给Native侧去直接管理这样做针对非首次打开这个场景是合适的,但是对首次打开这个场景却是不合适的。但是在首次打开这个场景下,onResume时Flutter页面尚未初始化,此时还不知道页面信息,因此也就不知道进入了什么页面,所以需要在Flutter页面初始化(init)时再回过来调Native侧的进入页面埋点接口。而为了避免开发人员去关注是否为首次打开Flutter页面,因此我们统一在Flutter侧来直接触发进入/离开页面事件。
曝光坑位
先讲下曝光坑位在我们这里的定义,我们认为图片和文本是有曝光意义的,其他用户看不见的是没有曝光意义的,在此之上,当一个坑位同时满足以下两点时才会被认为是一次有效曝光:
- 坑位在屏幕可见区域中的面积大于等于坑位整体面积的一半。
- 坑位在屏幕可见区域中停留超过500ms。
基于此定义,我们可以很快得出如下图所示的场景,在一个可以滚动的页面上有A、B、C、D共4个坑位。其中:
- 坑位A已经滑出了屏幕可见区域,即invisible;
- 坑位B即将向上从屏幕中可见区域滑出,即visible->invisible;
- 坑位C还在屏幕中央可视区域内,即visible;
- 坑位D即将滑入屏幕中可见区域,invisible->visible;
那么我们的问题就是如何算出坑位在屏幕内曝光面积的比例。要算出这个值,需要知道以下几个数值:
- 容器相对屏幕的偏移量
- 坑位相对容器的偏移量
- 坑位的位置和宽高
- 容器的位置和宽高
其中坑位和容器的宽和高很容易获取和计算,这里就不再累述。
获得容器相对屏幕的偏移量
//监听容器滚动,得到容器的偏移量
double _scrollContainerOffset = scrollNotification.metrics.pixels;
获得坑位相对屏幕的偏移量
//曝光坑位Widget的context
final RenderObject childRenderObject = context.findRenderObject();
final RenderAbstractViewport viewport = RenderAbstractViewport.of(childRenderObject);
if (viewport == null) {return;
}
if (!childRenderObject.attached) {return;
}
//曝光坑位在容器内的偏移量
final RevealedOffset offsetToRevealTop = viewport.getOffsetToReveal(childRenderObject, 0.0);
逻辑判断
if(当前坑位是invisible && 曝光比例 >= 0.5) {记录当前坑位是visible状态记录出现时间
}
else
if(当前坑位是visible && 曝光比例 <
0.5
) {记录当前坑位是invisible状态
if(当前时间-出现时间 > 500ms) {调用曝光埋点接口}
}
点击坑位
点击坑位埋点没什么难点,很容易就可以想到下面的方案:
效果
经过多轮迭代和优化,目前线上Flutter页面的埋点准确率已经达到100%,有力地支持了业务的分析和判断。同时这套方案让业务同学在做开发时,对于页面进入/离开、曝光坑位可以做到无感知,即不用关心何时去触发,做到了简单易用和无侵入性。
未来
此外,针对页面进入/离开这个场景,由于闲鱼是基于Flutter Boost混合栈的方案,因此我们的解决方案还不够通用。不过未来随着闲鱼上的Flutter页面越来越多,我们后续也会去实现基于Flutter原生的方案。
原文链接
本文为云栖社区原创内容,未经允许不得转载。
揭秘!如何用Flutter设计一个100%准确的埋点框架?相关推荐
- 如何用Java设计一个简单的窗口界面(学习中.1)
如何用Java设计一个简单的窗口界面 一.前言 二.简单了解 1.Swing简介 2.框架(frame) 3.层次 三.步骤 1.打开eclipse,依次创建项目,包,类. 2.代码 2.1最简单的可 ...
- 面试题之如何用Java设计一个自动售货机
如何用Java设计一个自动售货机程序是一个非常好的Java面试题.大多数情况会在面试比较senior的Java开发者的时候出现.在一个典型的代码面试中,你需要在一定的时间内根据对应的条件完成相关的代码 ...
- 为什么用redis做缓存而不是mybatis自带的缓存_如何用Java设计一个本地缓存,涨姿势了...
最近在看Mybatis的源码,刚好看到缓存这一块,Mybatis提供了一级缓存和二级缓存:一级缓存相对来说比较简单,功能比较齐全的是二级缓存,基本上满足了一个缓存该有的功能. 当然如果拿来和专门的缓存 ...
- 如何用SQL设计一个图书管理系统<纯SQL>
最近在某鱼上有小伙伴让我帮他设计一个图书管理系统的数据库,从建库到简单的数据库,现在写完了,分享给大家哦! 我们先来看看他的要求,如下图: 根据以上需求我们来编写我们的SQL语句: 1. 创建数据库用 ...
- 如何用Qt设计一个多文档文本编辑器
目录 前言 一.设计目标 二.效果展示 三.设计过程 1.设计思路 1.1文件的打开和新建 1.2设置字体和字号 1.3设置字型和颜色 1.4设置文字对齐撤销等 2.核心代码 总结 前言 学习了有关Q ...
- python 通讯录课程设计_如何用Python设计一个通讯录类?
直接上代码:一共三个文件 CommunicateClass.py # @File : CommunicateClass.py class Communicate(): ""&quo ...
- 如何用 Netty 设计一个百万级推送服务?
1. 背景 1.1. 话题来源 最近很多从事移动互联网和物联网开发的同学给我发邮件或者微博私信我,咨询推送服务相关的问题.问题五花八门,在帮助大家答疑解惑的过程中,我也对问题进行了总结,大概可以归纳为 ...
- 如何用redis设计一个运动步数排行榜?
文章目录 一.背景 二.情景再现 三. 设计当天排行榜 redis 有序集合 sorted set用法详解 1. 添加member成员和score 2. 按照score的升序或降序遍历指定范围的展me ...
- 如何用Java设计一个简单的窗口界面(初级二)
如何添加组件 一.准备 1.这里介绍的是eclipse 2.常用组件的了解 3.常用布局 1.BorderLayout 布局 2.FlowLayout 布局 二.代码 1.简单的 2.构造中间容器,顺 ...
最新文章
- yolov3 pytorch错误集合
- make -j 的并行任务个数选择
- 如何删除SQL Server下注册的服务器
- vmstat命令列出的属性详解
- 开源 微软 语音识别_能用嘴,绝不动手!支持跨屏的语音输入法,它来了!
- 机器学习在企业管理中如何落地?25个行业近500名CIO这样说
- Mysql的key_len计算方法
- 我的家乡主题网页设计
- 如何通过W3school学习JS/如何使用W3school的JS参考手册
- 5试点城市BIM/CIM平台情况盘点
- 导入FontForge生成字体
- Unity问题(1)——mesh法线反转
- 摩尔庄园服务器显示不出,摩尔庄园电脑为什么玩不了 摩尔庄园电脑玩不了解决方案...
- Opengl三视图的坐标变换
- Android8.0 WIFI ap Tethering 相关知识
- 华硕 PRIME Z490-PLUS+i7-10700K黑苹果EFI引导文件
- 不要让Microsoft edge 打开IE浏览器的设置(兼容性问题)
- 智能座舱全舱感知系统SCSS
- win10安装开启telnet服务及使用
- 用python读取txt文件中的数据并画各类图形展示_Python实现读取txt文件中的数据并绘制出图形操作示例...
热门文章
- oracle执行sql痕迹,Oracle 查询刚执行的SQL
- python 粒子动画_python-盒子中有很多粒子-物理模拟
- php 递归格式化数组,利用php递归实现无限分类 格式化数组的详解
- tensorflow gpu安装_tensorflow-gpu安装配置
- java字符流实际上也是字节,[Java教程]Java字节流与字符流的区别
- 鸿蒙系统手机9月11日,鸿蒙系统9月11日,将有望正式成为国际第三大手机操作生态系统...
- matlab 白色像素点,MATLAB 簡單的計算白色輪廓中像素點的個數
- linux 多域名访问数据库,Linux下虚拟域名的实现
- 计算机专业英语宋,机电一体化专业英语宋主民章.pdf
- mysql xdevapi_MySql Connector/C++8简介