浏览器版本低!无法浏览完整内容,建议升级或更换浏览器。
简介

Android系统对应用长时间运行的要求日趋严格,轨迹追踪应用可能会被系统杀死,导致轨迹上传不完整。我们梳理了有助于Android APP保活方法,供开发者参考。

Android 8.0及以后版本保活建议

方法一:开发者自行向华为申请加入后台APP白名单,加入白名单后可具体申请方法可详询华为客服。以下提供过往开发者的经验供参考,但最终解释权为华为所有:https://github.com/852172891/Android-/tree/master

方法二:Android 8.0系统为实现降低功耗,会对后台应用获取用户当前位置的频率进行限制,只允许后台应用每小时接收几次位置更新。为了适配这一系统特性,开发者可以参考鹰眼Demo,生成一个前台服务通知,使得应用退到后台的时候,仍有前台通知在,从而规避Android 8.0系统对后台定位的限制。

核心代码如下:
Notification.Builder builder = new Notification.Builder(this);
Intent notificationIntent = new Intent(this, TracingActivity.class);
Bitmap icon = BitmapFactory.decodeResource(this.getResources(),
        R.mipmap.icon_tracing);
NotificationManager notificationManager = (NotificationManager)
getSystemService(Context.NOTIFICATION_SERVICE);
// PendingIntent
builder.setContentIntent(PendingIntent.getActivity(this, 0,
notificationIntent, 0))
        .setLargeIcon(icon)  // ()
        .setContentTitle("") //
        .setSmallIcon(R.mipmap.icon_tracing) //
        .setContentText("...") //
        .setWhen(System.currentTimeMillis());
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O && null !=
notificationManager) {
    NotificationChannel notificationChannel =
            new NotificationChannel("trace", "trace_channel",
                    NotificationManager.IMPORTANCE_HIGH);
    notificationManager.createNotificationChannel(notificationChannel);
    builder.setChannelId("trace"); // Android OchannelId
}
notification = builder.build(); // Notification
notification.defaults = Notification.DEFAULT_SOUND; //
LBSTraceClient mClient = new LBSTraceClient(mContext);
Trace mTrace = new Trace(serviceId, entityName);
mTrace.setNotification(notification);

通过上述方式开启前台服务后无需开发者手动释放,鹰眼SDK内部会随着停止鹰眼服务一并释放。开启前台服务在其他版本Android系统中使用也可 最大程度的增加定位进程的存活率,提升后台定位效果。

注意:在Android 8.0之后使用前台服务,需要申请以下权限:

<span style="color: #FF0000;">
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
</span>


Android 10.0系统对定位限制更新严格,新增了后台定位权限,如果应用退到后台仍需定位,需要申请以下权限:

<uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION" />
如何监控鹰眼service是否存活

判断鹰眼servcie存活有两种方法:

调用startTrace()来判断。若返回"服务已开启",则表明服务正在运行;若返回"success",则表明之前服务未在运行,但此时已被开启;若返回"服务正在开启",则表明之前服务未在运行,现在正在开启过程中,需等待服务开启完毕。

调用手机系统API判断当前正在运行的服务中是否有鹰眼服务。

维持鹰眼服务的运行

通常使用轨迹追踪的应用都需要长时间(后台)运行,开发者APP可尝试使用以下方式降低APP被杀死的概率:

关闭省电模式

若手机开启了省电模式、各手机厂商定制的优化模式(如小米手机的神隐模式)、第三方安全应用的省电模式(如360手机卫士),则进程可能会被杀死。 处理策略为,用户在手机端关闭省电模式,或将APP加入这些模式的白名单。

开启新线程定时开启鹰眼服务

当手机系统内存不足时,会回收进程,直至内存充足时系统会自动重启进程。针对这种情况,可在APP内新增一个独立的进程B,定时查询鹰眼serviceA是否是否出于开启状态。当监听到鹰眼进程A被杀死,则重新开启鹰眼服务。但如果进程B同时也被系统回收,则该方法失效。

设置APP允许后台运行

出于功耗和性能问题,当前Android允许禁止APP后台运行,即所谓的黑白名单。被禁止的APP是无法在后台运行,如果集成鹰眼SDK的APP需要后台继续运行,则需要设置允许后台运行。设置项大多手机都在设置功能的隐私和安全中提供。

保持屏幕常亮

手机(设备)在 锁屏状态下,Android系统会先进入浅休眠,然后深度休眠。APP会被挂起,导致继承鹰眼SDK的APP无法运行。设置手机(设备)保持屏幕常亮,可以防止休眠,保证APP运行。

保持手机处于充电状态

Android系统中电池功耗逻辑,会在设备处于充电状态时,不限制某些高功耗的功能,诸如网络链接,结合“保持屏幕常亮”,可以使集成鹰眼SDK的APP保活。

启用多媒体锁

为了确保MediaPlayer的承载的服务在系统睡眠的时候继续正常执行下去。Android为我们提供了一种唤醒锁(wake locks)的机制。它能够在系统睡眠时,依旧保持锁定硬件的正常工作。基于这种思路,可以在集成鹰眼SDK的APP中,使用Service继承MediaPlayer,播放一段无声音频文件,达到保活效果。确保在MediaPlayer执行的时候,哪怕系统睡眠了CUP也能正常执行。需要使用MediaPlayer.setWakeMode()为MediaPlayer设定唤醒锁。以下是setWakMode()的定义:setWakeMode(Context context, int mode)第一个參数是当前上下文,第二个參数为须要加锁的状态,被设定为int类型的常量,定义在PowerManager这个final类中。在这里仅仅须要设定为PARTIAL_WAKE_LOCK就可以。

// 设定CUP锁定 mediaPlayer = new MediaPlayer(); mediaPlayer.setWakeMode(getApplicationContext(), PowerManager.PARTIAL_WAKE_LOCK);

一般对于锁而言。锁定了通常须要解锁。可是这里的唤醒锁与MediaPlayer关联,所以仅仅须要在使用完之后release()释放MediaPlayer就可以,无需显式的为其解锁。在使用setWakeMode设定唤醒锁的时候,还必须为应用赋予对应的权限: <uses-permission android:name="android.permission.WAKE_LOCK"/>

APP加入系统白名单

Doze模式是Android6.0上新出的一种模式,是一种全新的、低能耗的状态,在后台只有部分任务允许运行,其他都被强制停止。当用户一段时间没有使用手机的时候,Doze模式通过延缓app后台的CPU和网络活动减少电量的消耗。若手机厂商生产的定制机型中使用到该模式,需要申请将app添加进白名单,可尽量帮助鹰眼服务在后台持续运行,代码如下:

//位置采集周期
// 在Android 6.0及以上系统,若定制手机使用到doze模式,请求将应用添加到白名单。
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
    String packageName = trackApp.getPackageName();
    boolean isIgnoring = powerManager.isIgnoringBatteryOptimizations(packageName);
    if (!isIgnoring) {
        Intent intent = new Intent(
Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS);
        intent.setData(Uri.parse("package:" + packageName));
        try {
            startActivity(intent);
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
}

清单文件中添加权限声明:

//位置采集周期
<uses-permission android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS"/>
保持设定的轨迹采集频率

手机锁屏后一段时间,cpu可能会进入休眠模式,此时无法严格按照采集周期获取定位依据,导致轨迹点缺失。避免这种情况的方式是APP持有电量锁。

退出应用时鹰眼服务存活问题

用户按下back键退出应用时,开发者可在MainActivity的onDestroy()方法中决定是否要继续追踪轨迹。

退出应用时,停止追踪轨迹:在onDestroy()方法中调用stopTrace(),此时无论是否联网,stopTrace()执行完毕后,鹰眼进程都将被被停止,不再追踪轨迹。

退出应用时,继续追踪轨迹:无需在onDestroy()中执行任何操作,由于activity和鹰眼服务的独立性,在退出应用时,仅会销毁Activity,并不会自动销毁鹰眼Service。因此,此时应用虽然退出了,但鹰眼服务仍在追踪轨迹。同理,当在设备的应用管理中,将正在运行的轨迹追踪应用停止,此时鹰眼service并不会停止,仍在后台运行
stopTrace后缓存数据上传问题

调用stopTrace后,将立即停止采集轨迹,对于缓存数据此时有两种情况:

设备网络通信正常:将继续回传缓存的轨迹数据,直至数据回传完毕,鹰眼服务停止,返回回调信息。数据回传过程中,如果遇到网络中断或鹰眼服务被杀死,将会中断回传,但数据仍会存储在设备端,待下次StartTrace时,优先回传缓存数据,再回传新采集的数据。
设备网络通信中断:立即停止采集轨迹,同时将缓存数据会存储在设备数据库中,鹰眼服务停止,并返回回调信息。

  • 文档根本没法用

  • 文档水平很差

  • 文档水平一般

  • 文档不错

  • 文档写的很好

如发现文档错误,或对此文档有更好的建议,请在下方反馈。问题咨询请前往反馈平台提交工单咨询。

提交反馈

拖动标注工具

添加矩形标注

添加箭头标注

完成

取消