如何提高推送的到达率

作者: 蚊子Skeeter | 来源:发表于2017-01-11 11:04 被阅读3146次

为了提高用户体验,现在大多数的应用都会增加推送功能,目前主流的第三方推送有 个推、mi push、百度、Jpush、极光等,但是推送的到达率却是不尽人意的,拿个推而言,服务器这边统计的结果是到达率仅有90%(仅做参考)。当然了还有官方的推送Google Cloud Messaging,可惜在国内然并卵,暂不做讨论。

推送到达率问题的解决是刻不容缓的,因为在目前互联网大用户量的场景下,10%的用户数还是相当大的。

原因

我们知道,推送的技术原理主要是保持网络的长连接,在TCP长连接建立成功的基础上,推送不能如期到达的原因主要和网络状况有关,比如网络慢、丢包等等,这个是所有网络访问遇到的问题,不是导致推送到达率如此低的主要原因。

那么,其最主要原因是什么呢?显然是TCP长连接持续保持这个前提未能得到保证,也就是:
推送时,移动端未在线

解决方案

现在我们找到了其原因所在,那么要解决这个问题,就要从两方面入手:

  • “不择手段”的保证移动端在线,保证TCP长连接持续建立
  • 缓存推送消息,用户上线后重新发送

保证移动端在线

其实也就是我们常说的进程保活,可以创建一个幽灵进程进行保活操作,也可以直接用应用主进程进行保活,用这个进程中建立TCP连接,保证其存活的最大时长。方案主要以下几种:

  • 利用系统广播拉起应用,包括系统广播和同系列应用广播
  • 启动前台service,由于我们不想让用户感知到,所以应利用系统漏洞取消通知栏Notification的显示

注意:
保证移动端在线确实能有效的提高推送的到达率,但是需要注意频繁的唤醒应用会导致应用耗电量的增加,所以要把握一定的度。

广播唤醒

利用系统广播

监听系统事件广播来唤醒应用,常用的广播有:

  • 开机,ACTION_BOOT_COMPLETED
  • 亮屏,ACTION_SCREEN_ON
  • 灭屏,ACTION_SCREEN_OFF
  • 插拔有线耳机,ACTION_HEADSET_PLUG
  • 电量充足,ACTION_BATTERY_OK

注意:

  • 部分机型可能对开机广播做了限制,所以可能收不到开机广播
  • ACTION_SCREEN_ONACTION_SCREEN_OFFACTION_HEADSET_PLUG广播只能在代码里注册,当app完全退出后就收不到这个广播了

不同的app进程,用广播相互唤醒

  • 嵌入第三方SDK会唤醒相应的app进程,比如
    • 微信的SDK会唤醒微信应用,支付宝支付的SDK会唤醒支付宝SDK
    • 个推SDK会唤醒其他嵌入个推SDK的应用
  • App会唤醒同公司的其他app,比如:支付宝、天猫、淘宝、UC等阿里系的应用,打开其中一款就有可能顺便唤醒其他几款应用

前台service

该方案是应用范围最最广泛的一种手段,主要是启动一个前台service,并利用系统漏洞避免其在通知栏处显示Notification。这样既能保证进程的优先级高于普通后台进程,又将用户感知降到最低。

思路:

  • API < 18时,启动前台Service时直接传入new Notification()
  • API >= 18,同时启动两个id相同并传入new Notification()的前台Service,然后再将后启动的Service做stop处理
public class DaemonService extends Service {
    private static final int DAEMON_SERVICE_ID = 123456789;
    private static boolean mAlive = false;

    @Override
    public void onCreate() {
        super.onCreate();
        mAlive = true;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN_MR2) {
            // API < 18时,直接传入new Notification()
            startForeground(DAEMON_SERVICE_ID, new Notification());
        } else {
            // API >= 18时,启动两个id相同的service,然后将后startForeground的service stopForeground/stop
            startService(new Intent(this, DaemonInnerService.class));
            startForeground(DAEMON_SERVICE_ID, new Notification());
        }

        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        mAlive = false;
    }

    public static boolean isAlive() {
        return mAlive;
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    /**
     * 用于API >= 18时灰色保活Service
     */
    public static class DaemonInnerService extends Service {

        @Override
        public int onStartCommand(Intent intent, int flags, int startId) {
            startForeground(DAEMON_SERVICE_ID, new Notification());
            stopForeground(true);
            stopSelf();
            return super.onStartCommand(intent, flags, startId);
        }

        @Nullable
        @Override
        public IBinder onBind(Intent intent) {
            return null;
        }
    }
}

adb shell dumpsys activity services 查看结果看到前台service已经启动,但在通知栏里并未显示

service状态.png

当然了,我们可以结合上面这两个方案:
创建一个广播DaemonReceiver,该广播监听某些系统事件广播,在广播处理中启动DaemonService

public class DaemonReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        startDaemonService(context);
    }
    
     private void startDaemonService(Context context) {
        if (DaemonService.isAlive()) {
            return;
        }

        Intent serviceIntent = new Intent(context, DaemonService.class);
        context.startService(serviceIntent);
    }
}

鉴于当app被杀死后是监听不到系统广播的,而我们还需要保持DaemonService以确保推送TCP连接的建立,那我们可以在DaemonServiceonDestroy()中启动一个新的service DaemonReStartService, 在DaemonReStartService中来重新启动DaemonService

Android中的应用就是这么一步步被玩的卡的不要不要的,所以请谨慎使用。

缓存推送消息

流程如下:

  • 客户端
    • 收到推送后,发送回执消息给服务器,并存储到本地数据库
    • 收到的推送消息的消息ID已存储到数据库中时,不做处理并重发回执消息
  • 服务器
    • 如果客户端未在线,则将该条消息保存到数据库
    • 在客户端上线后,取出推送消息发给客户端并标记为已删除
    • 在取消息时注意同种类消息是否需要合并,考虑时效性只保留一定时间内的推送消息,合并或者超时后标记为已删除
    • 推送时在数据库里保存记录,收到客户端回执后将该条消息标记为已删除,超时未收到回执消息则重发消息

总结

不以用户利益为出发点的手段都是耍流氓。
进程保活必定导致应用一直保持唤醒状态一直在后台运行,不可避免的导致耗电量增加;发送回执消息则会额外消耗用户流量(可以考虑一段时间内的回执消息合并后统一发送),服务器保存每条推送记录可能会导致服务器压力过大。
所以,在尽可能保证用户到达率的情况下,也要考虑节能和流量,和使用设计模式一样,凡事皆有度,万事不可过。

相关文章

  • 如何提高推送的到达率

    为了提高用户体验,现在大多数的应用都会增加推送功能,目前主流的第三方推送有 个推、mi push、百度、Jpush...

  • 厂商 push 不通排查指南

    为了提升「MPS 消息推送」的推送的到达率,mPaaS 集成了华为、小米等厂商的推送功能,从而有效地提高用户留存率...

  • 推送原理以及提高推送到达率的方案介绍

    何为推送? ​ 何为推送?去百度了一番,本想找个官方定义,奈何没有相中的。最后自己总结了一下:服务器将信息定向...

  • 2018 iOS面试题

    为了方便阅读,这里就不放一些常见的面试题了 1,如何设计移动端路由 2,怎么提高消息推送的到达率 3,怎么统计A...

  • 融云IM(二)-----提高推送到达率

    这个都是可以直接看融云文档,我也是按照融云文档接,也没什么好的方法要提高推送到达率必须接第三方的推送,融云有进一步...

  • Android面试题汇总

    面试题 单例模式实现方式,优缺点? 如何提升推送到达率? 如何减小Apk体积? 如何优化网络请求? ListVie...

  • Android推送-在已有基础上如何提高到达率

    因为安卓app推送经常无法收到或者无法按时收到,所以公司要求我写一个针对推送的优化文档,为后面的优化做指导。我们当...

  • iOS怎么推送统计到达率

    掘金地址github地址 ios摘要 iOS10里的通知与推送 国内 90%以上的 iOS 开发者,对 APNs ...

  • 【产品】20分钟课程学习(160-164课)

    用户运营的用户召回 - 邮件:成本低、打开率低 - 推送:成本低、可能导致用户卸载APP - 短信:到达率...

  • Cocos Unity安卓接入友盟推送

    项目原来使用百度云推送,推送到达率比较低,了解一下友盟推送。 友盟入口:https://www.umeng.com...

网友评论

  • 小小Y2012:你好,请问:
    关于缓存推送消息,回执是先由产商服务端发给客户端,再由客户端传给自己的服务端吧?
  • 文武锅:这些办法早都出现了,依旧没卵用,该杀还得杀
  • 韦东锏:这些保活方案,现在主流的推送服务,都是实际在使用的吗?
    不同的app进程,用广播相互唤醒,这个很多机型有做了限制,避免被互相唤醒
    蚊子Skeeter:@韦东锏 大部分都是使用的前台service。系统广播确实被做了很大限制,但是不同app间的广播唤醒还是实现的

本文标题:如何提高推送的到达率

本文链接:https://www.haomeiwen.com/subject/iealbttx.html