神策(Android)- 在曝光采集基础上学习项目架构

news2024/11/16 1:54:14

开篇的时候我就在想这篇blog到底有没有意义?因为本身使用的就是神策提供的功能,同时神策也提供了很完善的文档,而唯一要我们做的也仅仅是将它正确的集成到项目内,并且随着版本升级,文档肯定也会有一定变更… 不过,当我通过曝光采集的集成来学习部分项目架构的时候,我感觉到了一些成长

什么是曝光采集?

指的应该是某些视图显示在当前页面时会进行数据采集,不同于以往的一些事件埋点;这种采集方式在用户滑动列表时也可以和便捷的采集到数据,正好与常规埋点互补

当你要集成 曝光采集 时,你肯定已经完成了神策的基础集成,所以我们直接进入 Android曝光采集 的接入流程~

一切以 官方文档 为准,因为随着版本升级,集成文档或许多多少少会有一些变动,此篇仅记录我集成曝光采集的过程

    • 基础配置
    • 常用功能
      • 普通、单视图标记
      • 列表元素标记
        • 标记所有行
        • 标记单行
      • 移除元素标记
      • 曝光回调
      • 更新某个元素的曝光属性
    • 项目实践
      • 基类封装
      • 基础配置
      • 调用封装

基础配置

 SAExposureConfig exposureConfig = new SAExposureConfig(areaRate, stayDuration, repeated);

SAExposureConfig 参数:只有当曝光配置 areaRatestayDurationrepeated 三者都满足时才能触发曝光埋点事件

在这里插入图片描述

示例

 //当我们设置元素完全曝光,有效停留时长 1s ,支持重复曝光
 SAExposureConfig exposureConfig = new SAExposureConfig(1.0f, 1, true);

SDK 提供全局设置曝光配置

 String SA_SERVER_URL = "数据接收地址";
 // 初始化 SDK 配置
 SAConfigOptions configOptions = new SAConfigOptions(SA_SERVER_URL);
 // 初始化曝光模块配置
 configOptions.setExposureConfig(exposureConfig);

常用功能

普通、单视图标记

官方介绍的方式肯定更简单、直白一些,在项目使用中肯定要进行二次封装,不然太乱了~

View view = findViewById(R.id.resourceID);
String event = "曝光事件名称";
JSONObject properties = new JSONObject();
try {
    properties.put("曝光事件属性 key", "曝光事件属性 value");
} catch (JSONException e) {
    e.printStackTrace();
}
SAExposureConfig exposureConfig = new SAExposureConfig(1.0f, 1, true);
SAExposureData exposureData = new SAExposureData(event, properties, exposureConfig);
SensorsDataAPI.sharedInstance().addExposureView(view, exposureData); //标记视图元素

SAExposureData 参数介绍

在这里插入图片描述
addExposureView 参数介绍

在这里插入图片描述
注意

  • 全局的曝光设置有一个默认值,即 areaRate = 0.0f,stayDuration = 0,repeated = true
  • 标记视图元素时,如果不传入配置项,则会使用全局曝光配置(可常用)
  • 标记视图元素时,如果传入配置项,则会使用传入的自定义配置(看情况使用,定制化强)
  • 不同的视图元素可以使用不同的曝光配置,曝光配置和标记视图时传入的配置对应

列表元素标记

当列表使用曝光的时候,由于列表元素在绘制过程中会复用,因此针对列表复用场景需要对列表元素进行唯一标识 exposureIdentifier 设置,一般可以通过 SAExposureData 进行设置,具体如下:

  SAExposureData exposureData = new SAExposureData("exposureName", null, "11",exposureConfig);

SAExposureData 参数介绍

在这里插入图片描述
注意:当元素(view)设置了唯一标识(exposureIdentifier),则通过唯一标识来进行曝光元素识别;因此不建议同一个页面不同元素设置相同唯一标识,只建议列表使用的时候进行元素唯一标识设置以避免列表复用导致的采集不准。

标记所有行

当针对列表中的某行或者某列进行标记而不是所有的行或列进行标记的时候需要通过 setExposureIdentifier 进行标记所有的元素,然后再通过 addExposureView 进行标记需要曝光的行或列的元素。

public void onBindViewHolder(@NonNull ViewHolder holder, final int position) {
     View view = holder.xxxx; 
     SAExposureConfig exposureConfig = new SAExposureConfig(1.0f, 1, true);
     SAExposureData exposureData = new SAExposureData("exposureName", null, String.valueOf(position), exposureConfig);//此处的曝光标识,一般传入需要标记的列表元素的位置
     SensorsDataAPI.sharedInstance().addExposureView(view, exposureData);//添加曝光元素(view)到曝光列表 
}

标记单行

  //进行标记所有的元素
  SensorsDataAPI.sharedInstance().setExposureIdentifier(view, String.valueOf(position));

setExposureIdentifier 参数介绍

在这里插入图片描述

public void onBindViewHolder(@NonNull ViewHolder holder, final int position) {
    View view = holder.xxxx;
    SensorsDataAPI.sharedInstance().setExposureIdentifier(view, String.valueOf(position));//曝光标识用于区分列表元素的特殊场景,比如列表元素复用、列表元素位置变更(刷新、删除、添加等),建议客户使用列表元素的位置作为唯一标识来处理
    //只采集列表 position 为 1 的曝光元素的曝光事件
	if(position==1){
        SAExposureConfig exposureConfig = new SAExposureConfig(1.0f, 1, true);
        SAExposureData exposureData = new SAExposureData("exposureName", null, String.valueOf(position),exposureConfig);//此处的曝光标识,一般传入需要标记的列表元素的位置
        SensorsDataAPI.sharedInstance().addExposureView(view, exposureData);//添加曝光元素(view)到曝光列表
    }
}

注意:如果标记某一行或某一列时,不使用曝光标识 setExposureIdentifier 进行设置,会导致曝光数据采集不准,单纯靠 position 是无法准确判定当前 position 是否是想要标记的元素;并且 setExposureIdentifieraddExposureView 进行标记的同一个元素 view 的唯一标识需要相同


移除元素标记

这个我在项目中没有使用到,具体看业务需求

当我们不想采集某个元素的曝光时,可以使用 SDK 提供的移除标记接口removeExposureView,有如下两个接口:

//方法一:
View view = findViewById(R.id.resourceID);
SensorsDataAPI.sharedInstance().removeExposureView(view);

//方法二:
View view = findViewById(R.id.resourceID);
SensorsDataAPI.sharedInstance().removeExposureView(view,"aaa");

removeExposureView 参数介绍
在这里插入图片描述


曝光回调

SDK 从 6.6.9 版本提供了一个曝光监听的协议接口, 可通过 SAExposureData 中的 exposureListener 来设置,接口说明如下:

saExposureData.setExposureListener(SAExposureListener saExposureListener)

SAExposureListener 说明如下

public interface SAExposureListener {
    /**
     * 返回对应 View 是否曝光
     *
     * @param view View
     * @param exposureData View 对应数据
     * @return true:曝光,false:不曝光
     */
    boolean shouldExposure(View view, SAExposureData exposureData);
 
    /**
     * 曝光完成回调
     *
     * @param view View
     * @param exposureData 曝光数据
     */
    void didExposure(View view, SAExposureData exposureData);
}

参数介绍

在这里插入图片描述

注意:shouldExposuredidExposure 回调都在主线程执行,禁止做耗时操作。

SAExposureData exposureData = new SAExposureData("expose");
exposureData.setExposureListener(new SAExposureListener() {
	@Override
	public boolean shouldExposure(View view, SAExposureData exposureData) {
		//可以根据实际业务来返回 true 或者 false,这里影响当前曝光监听者的所有曝光事件,需谨慎使用
		if (不需要曝光的条件) {
			return false;
		}
		return true;
	}

	@Override
	public void didExposure(View view, SAExposureData exposureData) {
		//可以根据实际业务需求,在触发曝光后,进行一些业务操作
	}
});
SensorsDataAPI.sharedInstance().addExposureView(view, exposureData);

更新某个元素的曝光属性

SDK 从 6.6.9 版本提供了一个曝光监听的协议接口, 可通过 updateExposureProperties 更改曝光元素的属性。

  SensorsDataAPI.sharedInstance().updateExposureProperties(view, properties);

参数介绍

在这里插入图片描述


项目实践

以下均为项目伪代码,未必可以直接使用,更多的提供一种借鉴、思想 ,有的东西我也未必全部理解,主要可以看看一些封装思想,而且采用了 Hilt 框架,关于这个框架我后续争取也记录一篇

已知的有几点要提示一下

  • 涉及多模块、组件化开发
  • 采用 DaggerHilt 依赖注入框架

基类封装

声明 BaseApplication ,主要提供获取上下文 和 资源获取的方法

import android.app.Application
import android.content.ContextWrapper
import android.content.res.Configuration
import android.content.res.Resources

abstract class BaseApplication : Application() {

    override fun onCreate() {
        super.onCreate()
        INSTANCE = this
    }

    @Suppress("DEPRECATION")
    override fun getResources(): Resources {
        val res = super.getResources()
        val config = Configuration()
        config.setToDefaults()
        res.updateConfiguration(config, res.displayMetrics)
        return res
    }
}

private lateinit var INSTANCE: Application

object AppContext : ContextWrapper(INSTANCE)

声明特用于神策的 LiuApplication ,正常应该是项目级Application

import dagger.hilt.android.HiltAndroidApp
import javax.inject.Inject

@HiltAndroidApp
class LiuApplication : BaseApplication() {

    @Inject
    lateinit var startups: Set<@JvmSuppressWildcards ApplicationStartup>

    override fun onCreate() {
        super.onCreate()
        // 注册多模块 onCreate()
        startups.sortedBy { it.priority() }.forEach {
            it.onCreate(this)
        }
    }
}

启动优先

import android.app.Application

interface ApplicationStartup : Priority {

    override fun priority(): Int = 100

    fun onCreate(application: Application)
}

优先级实体类

interface Priority {
    fun priority(): Int
}

基础配置

基础配置(已集成神策SDK基础配置 - Application

    private fun init(context: Context) {
        // 开启全埋点 其他配置,如开启可视化全埋点 需要在主线程初始化神策 SDK
        SensorsDataAPI.startWithConfigOptions(context, SAConfigOptions(serviceUrl).apply {
            // 开启全埋点
            autoTrackEventType = SensorsAnalyticsAutoTrackEventType.APP_START or
                    SensorsAnalyticsAutoTrackEventType.APP_END
            // 打开 SDK 的日志输出功能
            enableLog(BuildVariants.isDebug())
            // 开启 App 打通 H5
            enableJavaScriptBridge(true)
            // 传入 true 代表开启推送点击事件自动采集
            enableTrackPush(true)
        })
        trackAppInstall(context)

        //初始化一个曝光配置
        val exposureConfig = SAExposureConfig(1.0f, 1.0, true)
        // 初始化 SDK 配置 - 数据接收地址
        val configOptions = SAConfigOptions(serviceUrl)
        // 初始化曝光模块配置
        configOptions.exposureConfig = exposureConfig
    }

调用封装

建议观察顺序 ServiceManager - ServiceInterface - StatisticsService

StatisticsService 具体服务 - 埋点工具类(主要声明埋点方法,埋点所需参数等)

interface StatisticsService {

    companion object {
        @JvmStatic
        val service: StatisticsService by lazy { ServiceManager.queryStatisticsService() }
    }

    /**
     * 测试曝光
     */
    fun test(params1: String? = "", params2: String? = null)
 }

ServiceManager 服务管理类(偏代理)

object ServiceManager {

    private val serviceInterface by lazy {
        EntryPoints.get(AppContext.applicationContext, ServiceInterface::class.java)
    }

    fun queryStatisticsService(): StatisticsService = serviceInterface.bindsStatisticsService()
 }

ServiceInterface 通过 Hilt 生成对应服务类(偏工厂)

@EntryPoint
@InstallIn(SingletonComponent::class)
interface ServiceInterface {
    fun bindsStatisticsService(): StatisticsService
}

调用方式

 StatisticsService.service.test(params1,params2)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/700972.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

STM32微控制器:现状与竞争力的评估

STM32是意法半导体&#xff08;STMicroelectronics&#xff09;开发的一系列32位ARM Cortex-M微控制器。它们被广泛用于嵌入式系统开发&#xff0c;并在许多应用领域中得到了广泛应用&#xff0c;包括消费电子、工业自动化、汽车行业和物联网等。 尽管我无法提供最新的市场趋势…

(4)深度学习学习笔记-Softmax

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、来源 前言 softmax和cross-entorpy 一、 # softmax import torch from torch import nn from d2l import torch as d2lbatch_size64 train_iter,test_ite…

记录STM32使用udp通信的一个大坑

TOCfreeRTOSlwip实现udp通信 问题说明 在使用MCU和其他终端udp通信时遇见这样的一个大坑&#xff0c;整个通信过程如下图所示 问题出在mcu与其他设备通过udp交互&#xff0c;但在调试的过程中发现MCU给其他设备发消息的时候&#xff0c;虽然看起来成功了&#xff0c;但实际上…

尚硅谷Docker实战教程-笔记03【Docker常用命令】

尚硅谷大数据技术-教程-学习路线-笔记汇总表【课程资料下载】视频地址&#xff1a;尚硅谷Docker实战教程&#xff08;docker教程天花板&#xff09;_哔哩哔哩_bilibili 尚硅谷Docker实战教程-笔记01【理念简介、官网介绍、平台入门图解、平台架构图解】尚硅谷Docker实战教程-笔…

Vue+G6搭建力导向图

1 效果 初始化时节点向中间聚拢 拖动后&#xff0c;随引力作用缓缓向中间聚拢 点击节点&#xff0c;节点放大&#xff0c;并展示标签文字 2 代码分析 2.1 数据 2.1.1 节点数据 nodes: [{id: 0,label: 0,value: 10,cluster: a,description: this is node 0, \nand the value…

帆软数据填报——多字段联合校验数据是否重复

功能&#xff1a;同一张表中&#xff0c;在填报时&#xff0c;设定多个字段联合维度下&#xff0c;记录不允许出现2条及以上 EG&#xff1a;同个项目同个时间维度不允许存在2条及以上的记录 效果&#xff1a; 说明&#xff1a;如果设定“管理项目编码”和“统计截止日期”字段…

【树莓派】解密树莓派Python项目中神秘的导入错误

文章目录 问题导入python文件目录分析解决方案 问题导入 小编在使用树莓派编写python项目时出现了以下两种错误&#xff1a; ModuleNotFoundError : No module named Motor from ..hardware.motor import Motor portError: attempted relative import with no known parent p…

【C语言】手把手带你解决青蛙跳台阶问题

君兮_的个人主页 勤时当勉励 岁月不待人 C/C 游戏开发 Hello,这里是君兮_&#xff0c;今天更新的是经典递归问题——青蛙跳台阶&#xff0c;在所有有关递归的问题中&#xff0c;青蛙跳台阶是最广为人知的问题之一&#xff0c;可以说&#xff0c;如果你能真正弄懂青蛙跳台阶问…

javaweb学习2

p标签使用 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title> </head> <body> <!--p标签定义段落 p元素自动在其前后创建一段空白--> hello&#xff0c;world &l…

电脑鼠标怎么改变形状

电脑鼠标怎么改变形状? 首先你要下载一个鼠标指针包&#xff0c;网上有很多这样的资源&#xff0c;选择你喜欢的那一个指针包。 点击开始菜单&#xff0c;找到控制面板&#xff0c;打开控制面板。 从控制面板里点击“硬件和声音”&#xff0c;在设备和打印机里面点击“鼠标”。…

1-Eureka服务注册与发现以及Eureka集群搭建(实操型)

1-Eureka服务注册与发现以及Eureka集群搭建&#xff08;实操型&#xff09; 1. 简单搭建微服务框架1.1 idea创建maven多模块项目1.2 项目结构1.3 项目依赖与配置1.3.1 父工程&#xff1a;dog-cloud-parent1.3.2 管理实体项目&#xff1a;dog-po1.3.3 服务提供者&#xff1a;dog…

HBase(9):过滤器

1 简介 在HBase中,如果要对海量的数据来进行查询,此时基本的操作是比较无力的。此时,需要借助HBase中的高级语法——Filter来进行查询。Filter可以根据列簇、列、版本等条件来对数据进行过滤查询。因为在HBase中,主键、列、版本都是有序存储的,所以借助Filter,可以高效地…

主流开源深度学习框架简介

主流开源深度学习框架简介 本文目录&#xff1a; 一、TensorFlow深度学习框架 二、PyTorch深度学习框架 三、Keras深度学习框架 四、Caffe深度学习框架 五、中国深度学习开源框架状况 六、几种框架的对比 七、其他统计数据 当下&#xff0c;有许多主流的开源深度学习框架…

mysql 模糊查询的字段 支持不区分大小写功能

1. 直接修改字段对应的校对规则即可 ​​​​​​ 2. 校对规则说明 ​ utf8_bin 将字符串中的每一个字符用二进制数据存储&#xff0c;区分大小写。 utf8_genera_ci 不区分大小写&#xff0c;ci为case insensitive的缩写&#xff0c;即大小写不敏感。utf8_general_cs 区分…

香蕉派BPI-R4 Wifi7路由器采用联发科MT7988A (Filogic 880)设计

香蕉派BPI-R4路由器板采用联发科MT7988A (Filogic 880)四核ARM Corex-A73方案设计&#xff0c;板载4GB DDR4内存,8GB eMMC存储,128MB SPI-NAND闪存&#xff0c;还具有2个10Gbe SFP光电口, 4x Gbe千兆网口&#xff0c;带USB3.2端口&#xff0c;M.2接口支持4G/5G/NVME SSD.2x min…

海外版“咸鱼”Carousell是什么?

做跨境的都知道&#xff0c;一定不能只在一颗树上吊死&#xff0c;潮流总是把你推着向前的&#xff0c;现在跨境电商平台一样层出不穷。今天就来跟大家分享最近发现比价好做的外贸电商平台APP——Carousell。号称海外版“咸鱼”&#xff0c;它的功能也与咸鱼是差不多的&#xf…

群晖ssh登录并开启root用户登录功能

说明 群辉默认不允许SSH使用ROOT远程登陆&#xff0c;只能用手动创建的账户登陆&#xff0c;需要拷贝破解文件的时候会提示权限不足&#xff0c;这个时候肯定需要通过root用户去登录靠北 启动ssh登录功能 双击控制面板 点击终端和SNMP 启用ssh和telnet 点击应用 finalshel…

SpringBoot入门:使用IDEA构建第一个SpringBoot项目

SpringBoot框架介绍 Spring Boot是一个简化Spring开发的框架&#xff0c;用来监护spring应用开发&#xff0c;约定大于配置&#xff0c;去繁就简&#xff0c;just run 就能创建一个独立的&#xff0c;产品级的应用。我们在使用Spring Boot时只需要配置相应的Spring Boot就可以…

编译原理三:词法分析

概念 在编译器工作流程中&#xff0c;词法分析是将源代码分解为一系列词法单元的过程。 词法单元包括标识符、关键字、运算符等。词法分析器会读取源代码的每一个字符&#xff0c;根据预定义的规则将它们组成一系列词法单元。 词法分析器通常使用有限状态机来实现。有限状态机…

如何解决在vue3+vite项目中加载不了.tga文件的问题

因项目需求所以需要加载 .tga 类型的文件 import backgroundImage from "/assets/combined_image.tga" import { TGALoader } from "three/addons/loaders/TGALoader.js";const loader new TGALoader(); const texture1 loader.load(backgroundImage);但…