用户触达:可以简单理解为通过某种方式将消息传递给用户的行为;触达的特定消息从功能上可分展示、引导落地两层。用户触达作为一种产品运营方式,已经融入我们日常生产活动的方方面面。在移动互联网的世界里,我们的产品离不开触达,用户活动也离不开触达。
以用户使用角度来看,用户在使用 App 的过程中会有一些与用户相关的系统类的通知,比如交易物流、客服消息、账单信息,借还款提醒,实时资讯等消息需要及时的给用户提醒;
以 APP 运营活动看,App 在日常运营过程中,根据当前的目标,结合活动向用户定向发送相关营销类信息,比如单品的活动信息或一些品类促销优惠等,引导用户快速进入活动页面;
因此触达在拉新、促活、留存、变现、自传播等运营活动中扮演者重要角色。这篇文章从 app 研发视角介绍下用户触达方面的一些实践。
从 APP 的存活状态区分,实现触达有两种方式,一种是:APP 非活跃状态时的站外触达,主要包含:短信、Push、桌面小组件等
另一种是:APP 活跃状态时的站内触达,主要包含站内弹窗、页面固定运营位,feed 流推荐位等。
下面介绍下一下我们实现的几种触达方式及遇到的一些问题。
短信起初应用最广泛的场景是作为我们交流沟通的一种方式,随着时代的发展微信、QQ 等即时通讯类的 app 逐渐代替了短信作为人与人沟通工具,但是由于短信能够及时稳定的将消息同步给用户的特点,它仍是我们现在使用比较广泛的消息触达方式。常见的应用场景如:验证码通知、还款提醒、账户变动、营销活动通知等。我们知道作为一种触达方式,它的使命不仅是将消息通知到用户,对于特定的消息还要能便捷的引导用户跳转到 APP 内的相应的落地页。
短信的消息触达能力是毋庸置疑的,虽然短信文本中直接放入的链接我们也可以打开,但是确存在一些局限性,这种方式仅支持打开 web 页面,无法跳转到 APP 原生页面,另外点击链接会先弹窗,由用户选择打开链接的 app,这种体验相比直接打开 APP 指定页面来说大打折扣。因此,如何通过短信直接到达 APP 内
相应的落地页就是我们需要解决的问题。google 提供了一种能使 Android 系统直接通过网站地址打开应用程序对应内容页面,而不需要用户选择使用哪个应用来处理网站地址的方式,即 Android App Links;其工作流程如下:
要添加 Android App Links 到应用中,需要在应用里定义通过 Http (s) 地址打开应用的 intent filter,并验证你确实拥有该应用和该网站。
如果系统成功验证到你拥有该网站,那么系统会直接把 URL 对应的 intent 路由到你的应用。
1. 在 AndroidManifest 里配置用于系统进行验证的 IntentFilter:
当 android:autoVerify="true" 出现在你任意一个 intent filter 里,在 Android 6.0 及以上的系统上安装应用的时候,会触发系统对 APP 里和 URL 有关的每一个域名的验证。验证过程设计以下步骤:
系统会检查所有包含以下特征的 intent filter:Action 为
android.intent.action.VIEW、Category 为
android.intent.category.BROWSABLE 和
android.intent.category.DEFAULT、Data scheme 为 http 或 https
2. 配置一个数字资产链接的 Json 文件,声明你的网址和应用之间的关系;
对于在上述 intent filter 里找到的每一个唯一的域名,Android 系统会到对应的域名下查找数字资产文件,地址是:https:// 域名
/.well-known/assetlinks.json
只有当系统为 AndroidManifest 里找到的每一个域名找到对应的数字资产文件,系统才会把你的应用设置为特定链接的默认处理器。
数字资产示例:
package_name:在 build.gradle 里定义的 application ID
sha256_cert_fingerprints:应用签名的 SHA256 指纹信息,这个字段支持多个指纹信息,可以用来支持不同的应用版本,如开发版本和发布版本然后将 assetlinks 发布到 https:// 域名
/.well-known/assetlinks.json
[
{
"relation": [
"delegate_permission/common.handle_all_urls"
],
"target": {
"namespace": "android_app",
"package_name": "xxx.xxx.xx",
"sha256_cert_fingerprints": [
"xx:xx...."
]
}
}
]
3. 跳转落地页
在配置了上述 intent filter 的 Activity 中解析 url,并执行跳转落地页等操作
4. 问题及排查方法
如果配置后点击短信的链接无法正常跳转,可以逐个排查相关配置是否正确
4.1 确认数字资产文件是否被正确地定义和发布:
https://digitalassetlinks.googleapis.com/v1/statements:list?
source.web.site=https:// 你的域名:可选的端口
&relation=delegate_permission/common.handle_all_urls
4.2 确认应用是否设置了正确链接处理方式:
adb shell am start -a android.intent.action.VIEW \
-c android.intent.category.BROWSABLE \
-d "http:// 你的域名:可选的端口"
4.3 检查链接策略
这一步需要在应用安装后,等待一段时间 10s 后再执行,因为应用安装后系统会请求解析配置表
执行:adb shell dumpsys package domain-preferred-apps 或 adb shell dumpsys package d
该命令返回了设备上每一个应用配置的列表,这个列表标明应用和网站之间的关联
App linkages for user 0:
Package: com.android.demo 代表应用包名
Domains: play.google.com market.android.com 网站域名,多个网站之间用空格分隔
Status: always : xxxx 表示应用在 Manifest 文件里的配置了 android:autoVerify="true" 状态为 always;后面的 xxxx 和验证是否成功无关,和系统中应用的配置记录有关;
4.4 解决机型兼容性适配问题
在实践过程中还发现各厂商的不同型号的设备上存在无法跳转到落地页的情况,经分析该机型上应用安装后系统请求解析配置表 assetlinks 过程失败,此时会使用系统默认浏览器打开落地页,落地页是 app 原生页面的无法跳转到落地页,对于需要登录的 web 页面,如果未在登录中心注册的也会跳转失败,并会重定向到 m.jd.com 。
解决方案:有问题的机型,使用统一下载页中转,下载页执行唤起 APP,APP 内处理跳转落地页逻辑。
1. 客户端推送方案
Google 为 Android 提供了 FCM 推送,但是因为网络服务等一些原因其可用性不佳;目前国内各厂商 rom 也都提供了免费的 push 推送接入能力,同短信比 push 由于其免费性极大地节约了触达成本。
同时国内也有一些三方推送服务供应商,我们结合京东金融自身业务特点,为了保障数据的安全性以及推送消息的服务质量,最终采取整合华为,小米,OPPO、ViVO、魅族各厂商推送能力与自建通道相结合的方案。
其中厂商推送特点:token 有效期内,用户杀死 app 可以接收到 push 消息;自建通道特点:app 启动后建立连接,接收 push 消息,杀死 APP 后收不到 push 消息,主要用于使用未适配的厂商设备如三星、努比亚等用户接收 push 消息。
各厂商在 push 方案的实现上大体相同(厂商 push 接入流程,下图以 MiPush 为例),在使用厂商推送的过程中我们也遇到了很多问题,因此了解了各厂商的特性是制定出良好的触达策略前提。
2. 厂商推送遇到的问题
2.1 push 通知消息是否可以个性化展示
通常情况下通知栏消息展示效果主要内容包括消息标题、摘要、应用图标和时间。客户端可以自行定义具体展示内容。
不同厂商如华为、OPPO、vivo、小米、魅族等通知栏样式存在一些不同综合对比如下表:
在通知展示的样式上,综合对比来看华为支持 inBox 的样式,OPPO 小米支持大图样式,可以通过这些特点定制出更有特色的通知展示形式来突出通知主题。
华为 inBox 样式:Inbox 样式将每行内容都当作独立的单行文本去展示。文本内容最多可展示 5 行,每行内容展示不了时后边自动添加 “...”
OPPO 小米支持大图样式:这种通知可以将更有吸引力的图片展示给用户
2.2 App 有很多业务推送通知,用户是否可以指定接收分类消息
随着 APP 的业务越来越复杂,应用的通知越来越多,给用户造成明显打扰;
用户只能全局屏蔽这个应用的全部通知,不能屏蔽部分,然后留下对自己有用的。
为了解决这个问题,Android 8.0 开始支持开发者给自己的通知分成若干类,然后允许用户单独屏蔽这个类别的通知。
需要进行 Channel 分类,添加新 Channel(以 MiPush 为例):
ChannelHelper channelHelper = new ChannelHelper(APP_SECRET);
ChannelInfo channelInfo = new ChannelInfo.Builder()
.channelId("id") //必填,通知类别的ID,长度不超过 200 字符
.channelName("name") //必填,通知类别的名称,长度不超过40字符
.channelDesc("desc") //可选,通知类别的描述,长度不超过300字符
.notifyType(0) //必填,通知的效果类型,仅支持0,即振动、提示音、led灯三种效果都无
.soundUrl("sound_url") //可选,通知的自定义铃声uri,格式介绍请参见 “4.1 自定义铃声”
.build();
Result result = channelHelper.addNewChannel(channelInfo, 1)
不同的 channel 在系统设置页通知设置中展示如下:
通过细分 push 通知的类别,增加通道数量可提高 push 消息在通知栏里的留存率;
同时用户可有更多选择,设置自己比较关注的类型消息,避免过多打扰,以提升用户体验。
2.3 如何指定推送方式或人群
各厂商推送方式支持方式如下:
2.3.1 基于 ReglD 的推送
RegID 为是推送 SDK 为每个设备上的每个 app 注册推送服务时生成的唯一标示。
当开发者需要给一个或多个具体的设备推送消息时,可以使用基于 RegID 的推送,将个性化的信息推送给指定的设备。这种方式适用于需要为每个用户订制个性化推送的场景。
2.3.2 基于 Alias 的推送
alias 是推送提供的一种个性化设定,开发者可以将用户在应用内的账号或其它用户唯一标识设定为用户设备 RegID 的别名,在推送中可以直接基于别名进行推送。
别名不仅方便开发者将推送与自有的账号系统进行关联,同时也避免了因需要保存设备 RegID 与自有帐号的对应关系而额外带来的开发和存储成本。
2.3.3 基于标签的推送
对应用下已订阅 push 的设置了标签的用户进行推送。在推送消息时,开发者可以结合每条消息的内容和目标用户人群,选择所对应的标签,完成请求后,push 推送服务会向所有
打上这一标签的用户发送该消息,从而满足定向推送的需求。并且提供标签管理功能。
2.3.4 小米通道 userAccount :最多可对应 20 台设备,单账号可登陆多台设备,给一个 userAccount 推送可同时有 20 台设备收到消息。
总结:将特定的推送消息通过特定的方式发送给比如不同的客户端版本、 不同地域、男女等的用户群体,或者通过给不同的用户群体打不同的标签的方式实现特性消息的推送,以达到更精细推送的目的。
2.4 OPPO、ViVO 触达成功率低,如何提升
触达数据接入数据看板后,经对比各厂商触达成功率发现 OPPO、ViVO 的触达率基本在 83%~86% 而小米华为通道触达成功率基本在 94%~98% 因此提升 OPPO、VIVO 通达的触达成功率是我们面临的又一问题
OPPO: 经排查发现影响 OPPO 触达率的主要因素为通知开关的状态:APP 仅在通知开关开启的情况下才能收到厂商的 Push 消息而 OPPO、一加通知开关在用户安装后默认关闭,因此收不到 Push 消息。
解决办法:前期主要是制定引导策略,在合适的时机检测通知开关状态,引导用户主动去设置页开启,后来经调研发现 OPPO 的 ColorOS 系统提供了一键开启通知开关的能力,后期使用引导一键开启方案,将 OPPO 通道的触达率提升到了 94% 左右;
VIVO: 与 OPPO 不同,VIVO 设备安装应用后通知开关是开启的,我们根据数仓提供的数据与厂商反馈的错误码分析,导致 VIVO 触达偏低主要因素为消息未进行分类而被限额。
vivo 通道消息类型分为两类 —— 按消息类型是否与用户强相关将消息分为 “运营消息” 和 “系统消息”,未接消息分类功能将导致所有消息默认为运营消息而受到频控限制,从而导致重要消息可能无法触达。
vivo 用户单应用每日运营消息接收条数上限 5 条,系统消息无限制。vivo 用户单应用接收条数限制以 “到达量” 是否超过 5 条为准,在发送时校验单用户是否到达 5 条,超限则计入管控量。
除 VIVO 外,华为、OPPO、小米对通知消息的数量 都有一定的限制,对于存在限额的厂商通道,将点击率高的个性化推送策略尽量安排在上午推送,可以保证优质推送内容的到达率;通过提高消息推送的额度,提高 push 消息的触达率。
2.5 如何增强未读消息提醒
可以在 App 桌面角标显示未读消息数,厂商 lunch app 和 push sdk 对此提供了相应的能力支持,用于增强提醒,各厂商的实现细节上有差异:
华为:角标未读数由服务端下发的 push 消息控制,开放了 api 供第三方应用设置角标未读数,移除通知栏消息角标数量不会变化。
小米:角标未读数等于厂商 push 通道(系统通知栏)收到的该 app 的未读通知数,开放 api 供第三方应用设置角标未读数。移除系统通知栏消息,角标数量相应减少。
oppo:支持红点,数字角标,角标未读数等于厂商 push 通道(系统通知栏)收到的该 app 的未读通知数。
vivo:桌面角标未读数开关默认关闭,需要用户手动开启才能使用,提供设置角标未读数的能力。
角标适配的问题及解决办法:
2.5.1 在小米系统上能展示通知数,但无法更新站内信数量。
解决方案:站内信和 push 打通,进入 app 时同步更新未读数。
2.5.2 在华为系统上无法显示 Push 数量,站内信数显示正常。
解决方案:华为推送服务提供了在服务端设置桌面角标 API 接口,第三方 app 可以在消息中封装角标参数。
2.5.3 vivo 手机上不支持显示角标未读数。
解决方案:更新 SDK 版本,接入角标能力
2.5.4 在 oppo 角标展示仅站内信数量。
push 功能在开通时可以申请圆点角标或数字角标、无角标三种形式,用户可以在通知设置中自主选择。
oppo push 支持的系统版本,目前支持 ColorOS3.1 及以上的系统的 OPPO 的机型,一加 5/5t 及以上机型,realme 所有机型(Android 8.0 以后的设备)。
2.5.5 其他:魅族手机未开放桌面角标设置。
1. 站内横幅方案介绍
已有的触达方式对用户实时行为产生的场景覆盖不够,而且这类场景较离线场景相比实时性更高,对用户来说相对更重要。针对这个情况,我们增加了对实时场景覆盖。
目的是将用户行为抽象成关系模型,当关系一侧的用户行为发生变更后触发对另一侧的触达,这种情况实时性更强而且和用户强相关,触达的消息点击和转化都比较高,也有利于增强用户粘性。
站内横幅整体设计概览
数据服务层:各业务模块负责采集用户行为数据,由 molo 侧将用户行为抽象关系模型,用户进入指定场景,触发对应场景触达策略,再经统一频控量控进行核验
传输层:基于 MQTT 协议的长链接实现的鹰眼自建通道,将通过核验的触达信号传递给下一流程
APP 基础能力层:为触达消息传输,流程监控提供基础能力
数据解析层:将传递过来的触达消息体解析,合法性校验,监控异常数据
视图控制层:进行触达消息模板视图创建,弹出方式识别,通过 ViewCore 给触达消息视图注入生命周期,出入场动画,声音震动提醒、展示动效等各种定制化属性
2. 京东金融 App 站内横栏应用场景
站内横栏功能上线后,为一批业务提供了有效的触达策略
3. 遇到的问题及解决思路
3.1 如何让横栏实现在 App 站内全局
全局弹窗这个实现起来相对容易,主要依赖注册的页面生命周期监听,利用 WindowManager 在离开页面时移除 view ,在进入新页面重新添加;
3.2 指定页面显示或指定页面不显示问题
指定页面的前提是能区分是哪个页面,分两种情况:
Web 页面,首先获取运营在鹰眼平台配置的指定的 Web 链接,再通过 APP 的 web 容器获取当前正在加的 web 页面的链接地址,两个地址进行匹配,需要注意本地取到的 url 里参数存在比运营配置多的情况,因此匹配时我们认为只要本地取到的参数包含配置的地址中的各参数即是匹配成功;
原生页面,方式一,路由地址匹配:我们首先取原生页面的路由信息,本地有路由信息根据路由地址去匹配,若原生页面无路由地址,需要进行适配
方式二,popClass 匹配:需要将原生页面的类路径录入到后台页面配置表进行维护,匹配时根据页面的类路径进行匹配
3.3 如何避免多个横栏消息时丢失问题
同时支持多个横栏,这里需要注意的是横栏信息同步问题,我们在创建横栏的时候给横栏创建了一个属性信息对象,每个横栏属性信息都有唯一的 key, 将横栏属性缓存起来,并给缓存设置最大阈值,达到阈值时最后一个横栏消失清除缓存信息
总结:站内横栏触达方案是我们在智能化触达方式中的一项探索,功能上线后,为白条,保险,财富,基金,分期等业务提供了一种更智能化的运营方式,触达消息触达成功率 98%,点击率达到 12%~16%,助力相关业务提升 40% 以上
AppWidget 又称小部件、小插件或微件。它是显示在 Launcher 上,能在 Logo 以外提供更多信息的一种特别的设计;它方便用户免于打开 App 即可直接查看信息和进行简单的交互。
1. 创建 AppWidget
总的来说分以下几个部分:
1.1 定义 AppWidgetProvider
创建一个 AppWidgetProvider 子类,并创建对应的 AppWidgetProviderInfo 配置文件
example_appwidget_info.xml,并在 manifest 声明
1.2 设置 appWidget 的基本属性
AppWidgetProviderInfo 定义了 widget 的基本特性,如应用微件的最小布局尺寸、应用微件的初始布局资源、应用微件的更新频率,以及(可选)在应用微件创建时启动的配置 Activity。您可以使用单个
1.3 绘制 widget 的布局
AppWidget 可以支持的布局如下(由于其底层是基于 RemoteViews 实现,支持的视图较少):
支持使用的 View 如下(不支持自定义 View):
1.4 配置 Configuration Activity
当应用 widget 使用配置 Activity 时,由该 Activity 负责在配置完成后对 app 的 widget 进行初始化
1.4.1 获取 widget id
1.4.2 执行应用微件配置
1.4.3 配置完成后,通过调用 getInstance(Context) 来获取 AppWidgetManager 的实例
1.4.4 通过调用 updateAppWidget(int, RemoteViews) 来使用 RemoteViews 布局更新应用微件
1.4.5. 最后,创建返回 Intent,为其设置 Activity 结果,然后结束该 Activity
1.4.6 设置预览图片
在选择创建 appWidget 时,展示给用户的描绘应用微件是什么样子的一张图片,未配置时默认展示 APP logo
1.4.7 配置 Service
请求集合中的特定项目时,RemoteViewsFactory 会为集合创建相应项目并将其作为 RemoteViews 对象返回。要在 appWidget 中添加集合视图,您必须实现 RemoteViewsService 和 RemoteViewsFactory。
详细参见官方 demo:
https://android.googlesource.com/platform/development/+/master/samples/StackWidget/src/com/example/android/stackwidget/StackWidgetService.java
1.4.8 设置点击事件
通常使用 setOnClickPendingIntent() 来设置对象的点击行为 - 例如,让按钮启动 Activity。但是,不允许对各个集合项目中的子视图使用此方法。如果要向集合中的各个项目添加点击行为,应改用 setOnClickFillInIntent()。这需要为集合视图设置待定 Intent 模板,然后通过 RemoteViewsFactory 在集合中的每个项目上设置填充 Intent。
2. 常见 App 的实现
金融 APP 的实现:
3. 小组件实践中的问题
3.1 如何裁剪图片圆角
一般在开发过程中使用 Glide 对图片进行裁剪,这里需要注意小组件里使用 Glide 与平常略有不同,因为拿不到对应的 View 视图,AppWidgetTarget 更适用于小组件加载图片场景,配合 MultiTransformation 可简便的实现图片圆角的剪裁
AppWidgetTarget appWidgetTarget = new AppWidgetTarget(context, ivViewId, views, mAppWidgetIds);
RequestOptions option = new RequestOptions()
.transform(new MultiTransformation<>(
new CenterCrop(),
new RoundedCorners(ToolUnit.dipToPx(mContext, connerDp))));
GlideApp.with(context)
.asBitmap()
.load(bgUrl)
.apply(option)
.diskCacheStrategy(DiskCacheStrategy.NONE)
.into(appWidgetTarget);
3.2 如何实现自定义字体
小组件本身是不支持自定义 view 的,若要实现支持自定义字体,可以通过 Canvas draw text 方式 给 text 设置字体样式,粗细、颜色 、背景等属性
3.3 处理点击响应延迟问题
通过广播形式 PendingIntent.getBroadcast 处理点击事件,在部分机型上存在延时,最长约 7s;可以使用 setOnClickPendingIntent 方式代替,需要在 app 的跳转中心处理对应的事件,如跳转落地页、埋点等
3.4 如何制定更新策略
系统为了避免小组件过多的占用资源,默认拒绝频繁更新,设置了最短更新时间为 30 分钟;这种默认的刷新方式不太适合交互类型的小组件,存在用户操作完以后页面状态不同步的问题,那这个问题如何解决呢?首先根据业务场景需要我们也可以把刷新分为两类:
实时性有一定要求的业务场景:比如新闻资讯类的
通过创建 Service 开启定时任务的方式,制定更新的时间间隔,比如 5 分钟执行一次更新任务;
用户交互类型的业务场景:比如签到,收积分、能量等;此类场景不要求频繁刷新数据,但需要配置合理的自动刷新时间,同时在用户操作后需要刷新页面;可以在 App 启动时注册 APP 内页面生命周期监听ActivityLifecycleCallbacks,实现判断 APP 前后台监听能力,监听应用进入后台时发送刷新小组件的广播,触发小组件的刷新;或者封装统一方法,提供给业务主动触发刷新对应的小组件的接口
本文主要分享了京东金融客户端技术团队对短信、push、站内横幅、小组件几种触达方式的探索实践过程以及遇到的问题和解决方案。希望能给在探索用户触达实现方案的同学提供一些思路;为了让用户在使用我们的产品的时候能有更好的体验,产品在运营过程中能高效的触达用户,京东科技技术团队在持续打磨已有方案的同时将继续探索和实践更加智能高效的触达方案。