Android O 行为变更适配方案

Nothing makes an android developer more crazy than a new version of Android.

Android O 在 2017 年 8 月已经正式发布了,Android P 的预览版也已推出,今年下半年也将正式发布。

由于公司的产品面向海外用户较多,大多都已经是 Android 最新版本,且 Google 明确提出:从 2018 年 8 月起,所有向 Google Play 提交的新应用都必须使用 API level 26 (Android 8.0) 及以上版本开发;2018 年 11 月起,所有 Google Play 的现有应用更新同样必须使用 API level 26 及以上版本。因此必须对 Android O 作全面的适配。

1. Android O 新特性

官方文档已经详细描述了 Android O 的新特性和 API,对开发者而言适配 Android O 需要关注以下行为变更:

  • 后台服务限制
  • 隐式广播限制
  • 后台位置限制
  • 通知渠道
  • 权限

2. 后台服务限制

2.1 什么是前台应用

满足以下任意条件的应用被视为处于前台:

  • 具有可见的 Activity
  • 具有前台服务
  • 另一个前台应用关联到该应用,如输入法、壁纸服务、语音服务等

如果以上条件均不满足,应用将被视为处于后台。

2.2 为什么要限制后台服务

后台服务(如网络下载、数据同步等)会消耗手机的内存和电量,影响性能。如果大量应用都开启了后台服务,会严重影响用户体验。

后台服务限制和隐式广播限制从根本上杜绝了后台应用异常消耗系统资源,有助于大幅度降低应用后台行为对设备体验的影响。

2.3 什么是后台服务限制

应用在后台期间保留其后台服务的能力将受到限制。如果应用处于后台时调用了 startService() 将会抛出 IllegalStateException,除非:

  • 应用已经处于前台,则可以调用 startService(),不会抛出 IllegalStateException,但一旦进入后台,后台应用将被置于一个临时白名单中,位于白名单中时,应用可以无限制地启动服务,其后台服务也可以运行。但这个时间窗一过,应用进入空闲状态,后台服务就会被销毁(Nexus 5X 8.0 系统上测试不到1分钟)
  • 启动前台服务
  • 绑定服务,即使应用处于后台也不受影响

如果 targetSdkVersion < 26,是否可以绕过这些限制?

不可以,即使 targetSdkVersion < 26,用户也可以在 Android O 的设备上选择开启这些限制。

2.4 解决方案

  • Job Scheduler
  • Foreground Service
  • Firebase Cloud Messaging and Temporary Service Whitelist

2.5 Job Scheduler

JobScheduler is smarter about when jobs should be run and can batch them together so that devices stay asleep as much as possible.

Google 在 Android 5.0 中引入 JobScheduler 来执行一些需要满足特定条件但不紧急的后台任务,利用 JobScheduler 来执行这些特殊的后台任务来减少电量的消耗。可以将其理解成定时任务,以替代 IntentService + AlarmManager

开发者可以设定需要执行的任务 JobService,以及任务执行的条件 JobInfoJobScheduler 会将任务加入到队列。在特定的条件满足时 Android 系统会去批量的执行所有应用的这些任务,而非对每个应用的每个任务单独处理。这样可以减少设备被唤醒的次数。

2.5.1 JobService

先来看一下 JobService, 它继承自 Service,除了 Service 的一些生命周期方法,又增加了 onStartJobonStopJob 来处理自定义任务。

public class MyJobService extends JobService {  
   
    @Override
    public boolean onStartJob(JobParameters params) {
        Log.i(TAG, "onStartJob");
        doJob(params);
        return true;
    }

    @Override
    public boolean onStopJob(JobParameters params) {
        Log.i(TAG, "onStopJob");
        // whether or not you would like JobScheduler to automatically retry your failed job.
        return false;
    }

    private void doJob(JobParameters params) {
        // I am on the main thread, so if you need to do background work,
        // be sure to start up an AsyncTask, Thread, or IntentService!
    }
}

使用 JobService 需要在清单里申请 android.permission.BIND_JOB_SERVICE 权限

<service android:name=".MyJobService"  
         android:permission="android.permission.BIND_JOB_SERVICE" />

需要注意的是,在 com.android.server.job.JobServiceContext 类中声明了 EXECUTING_TIMESLICE_MILLIS

/** Amount of time a job is allowed to execute for before being considered timed-out. */
private static final long EXECUTING_TIMESLICE_MILLIS = 10 * 60 * 1000;  // 10mins.

经过测试,不管应用是否处于前台,JobService 都不能无限期运行,有 10 分钟的超时时间,会自动销毁,在 Android L 上这个时间是 1 分钟。因此 Job Scheduler 适用于短耗时的后台任务,不适用于连续的长时间的后台服务。

2.5.2 使用 Job Scheduler

实施一个 Job 包含以下步骤:

  1. JobInfo:采用 Builder 模式,设置 Job 执行的条件和时机
  2. JobServiceService 的子类,Job 执行时的具体行为
  3. android.permission.BIND_JOB_SERVICEJobService 的子类需要授予该权限
  4. JobScheduler:将一个 Job 添加到工作队列中,调用 JobScheduler.enqueue() 或者 JobScheduler.schedule。当工作队列运行时,它可以将待定的工作从队列中剥离并处理(替代 IntentService)。
void scheduleJob() {
    ComponentName componentName = new ComponentName(this, MyService.class);
    JobInfo.Builder builder = new JobInfo.Builder(JOB_ID, componentName)
            .setMinimumLatency(2000)
            .setOverrideDeadline(5000)
            // ...
            .setRequiredNetworkType(JobInfo.NETWORK_TYPE_NONE);

    JobScheduler jobScheduler = (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);
    jobScheduler.schedule(builder.build());
}

具体使用可以参考 Scheduling jobs like a pro with JobScheduler

2.5.3 在 API 低于 21 使用 Job Scheduler

JobScheduler 是Google 在 API 21 引入 的,那么对于 API 小于 21 该如何处理。

  • minSdkVersion >= 21:JobScheduler
  • minSdkVersion < 21:Firebase JobDispatcher,API 和 JobScheduler 基本相似,但需要引入 Google Play Services,会增加 apk 的大小。
  • 如果 minSdkVersion < 21,可以做一层封装,业务方只调一套,也可以只用Firebase JobDispatcher 兼容,看业务方需求。

2.5.4 JobIntentService

JobIntentService 继承自 Service,可以用来简化处理任务。调用 JobIntentService.enqueueWork(),即可执行任务。它会根据 API 版本进一步执行

  • 在 Android O 及之后的设备上,调用 JobScheduler.enqueue()
  • 在 Android O 之前的设备上,调用 Context.startService()

JobIntentService 需要 android.permission.WAKE_LOCK 的权限

<uses-permission android:name=”android.permission.WAKE_LOCK” />

2.6 Foreground Service

Job Scheduler 只适用于短耗时的后台任务,如果需要在后台执行长期的任务,推荐使用 Foreground Service。这样应用会在通知栏展示进行中的通知,以告知用户你的应用正在运行后台任务。

在 Android O 之前,创建前台服务的方式通常是先创建一个后台服务,然后将该服务推到前台。

但对于 Android O,系统不允许后台应用创建后台服务。 因此,Android O 引入了一种全新的方法,即
ContextCompat.startForegroundService(),以在前台启动新服务。

  1. 调用 ContextCompat.startForegroundService() 可以创建一个前台服务,相当于创建一个后台服务并将它推到前台。

  2. 创建一个用户可见的 Notification

  3. 必须立即(在5秒内)调用该服务的 startForeground(id: Int, notification: Notification) 方法,否则将停止服务并抛出 android.app.RemoteServiceException: Context.startForegroundService() did not then call Service.startForeground() 异常。

2.7 Firebase Cloud Messaging and Temporary Service Whitelist

使用 FCM 需要引入不低于 10.2.1Google Play Services SDK

当应用出现以下情况时,该应用可以被暂时加入到一个白名单里,应用可以像跑前台服务一样跑后台服务:

  • 处理高优先级的 FCM 消息
  • 接收广播,如短信/彩信消息
  • 点击通知执行 PendingIntent

这种情况常用的一个场景是,我们的应用需要通过请求后台服务更新数据,可以给我们的应用发送一个高优先级的 FCM 消息,即使系统处于休眠状态,也能马上收到 FCM 消息,被加到白名单后,就可以启动一个后台服务来更新数据了。

通过以上情况启动的也必须是短耗时的任务,如 JobScheduler 或者 JobIntentService

2.8 解决方案

  • 对于短耗时的特定任务,采用 Job Scheduler
  • 对于需要长期执行的服务,采用 Foreground Service
  • 对于一些三方的服务,无法修改,如果没有适配 Android O,可以在启动这些服务前启动一个空的 Foreground Service,这样应用处于前台,就可以启动这些后台服务了。

如果在应用初始化后启动一个空的 Foreground Service,保证应用处于前台,则旧的 Service 都可以正常,修改量最小。但不推荐。

3. 隐式广播限制

3.1 为什么要限制隐式广播

如果在 AndroidManifest.xml 中随意地声明隐式广播,那么任何时候收到响应系统事件都会唤醒应用,即使应用当前处于休眠状态,导致异常消耗系统资源。

3.2 什么是隐式广播限制

应用无法在 AndroidManifest.xml 中声明隐式广播接收器,以获得绝大部分响应系统事件的后台能力。

显式广播依然能在 AndroidManifest.xml 中注册。

3.3 解决方案

  • Broadcast Whitelist
  • Scheduling Jobs
  • Dynamic Broadcasts

3.3.1 白名单

以下事件依然能在 AndroidManifest.xml 中声明隐式广播接收器,可以正常使用。

  • ACTION_LOCKED_BOOT_COMPLETED, ACTION_BOOT_COMPLETED
  • ACTION_USER_INITIALIZE
  • ACTION_LOCALE_CHANGED
  • ACTION_USB_ACCESSORY_ATTACHED, ACTION_USB_ACCESSORY_DETACHED, ACTION_USB_DEVICE_ATTACHED, ACTION_USB_DEVICE_DETACHED
  • ACTION_HEADSET_PLUG
  • ACTION_CONNECTION_STATE_CHANGED, ACTION_CONNECTION_STATE_CHANGED, ACTION_ACL_CONNECTED, ACTION_ACL_DISCONNECTED
  • ACTION_CARRIER_CONFIG_CHANGED
  • LOGIN_ACCOUNTS_CHANGED_ACTION
  • ACTION_PACKAGE_DATA_CLEARED
  • ACTION_PACKAGE_FULLY_REMOVED
  • ACTION_NEW_OUTGOING_CALL
  • ACTION_DEVICE_OWNER_CHANGED
  • ACTION_EVENT_REMINDER
  • ACTION_MEDIA_MOUNTED, ACTION_MEDIA_CHECKING, ACTION_MEDIA_UNMOUNTED, ACTION_MEDIA_EJECT, ACTION_MEDIA_UNMOUNTABLE
  • SMS_RECEIVED_ACTION, WAP_PUSH_RECEIVED_ACTION

3.3.2 JobScheduler

JobScheduler 可以用来执行一些需要满足特定条件的后台任务,如设备网络状态变化、设备充电状态变化、低电量等。因此大多数情况下,之前注册隐式广播的应用使用 JobScheduler 可以获得类似的功能。

3.3.3 动态注册广播

依然能用 Context.registerReceiver() 动态的注册隐式广播,不受影响。但务必在不需要的时候(如生命周期结束)调用 Context.unregisterReciever()

4. 后台位置限制

在 Android O 上,应用处于后台时降低了后台应用接收位置更新的频率,具体的位置行为和受影响的 API 可以查看官方文档

5. 通知渠道

所有通知的实现都需要提供通知渠道(Notification ChannelId),否则通知在 Android O 系统上无法正常展示,会弹 Toast 提示 Developer warning for package XXX,Failed to post notification on channel “null”.

if (Build.VERSION.SDK_INT>= Build.VERSION_CODES.O) {
    NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);

    NotificationChannelGroup group = new NotificationChannelGroup(GROUP_ID, GROUP_NAME);
    manager.createNotificationChannelGroup(group);
    
    NotificationChannel channel = new NotificationChannel(CHANNEL_ID, CHANNEL_NAME, NotificationManager.IMPORTANCE_DEFAULT);
    channel.setGroup(GROUP_ID);
    // ...
    manager.createNotificationChannel(channel);

    notification = new Notification.Builder(getApplicationContext(), CHANNEL_ID)
            // ...
            .build();
} else {
    notification = new NotificationCompat.Builder(getApplicationContext(), CHANNEL_ID)
            // ...
            .build();
}

6. 权限

  • Android O 之前,申请一个子权限(如写外部存储权限),会自动获取权限组中其他子权限(读外部存储权限)。组内其他子权限可以直接使用,无需申请。

  • Android O 修复了这个错误。在 Android O 上,申请一个子权限,组内其他子权限不会自动获取,需要再次申请才能使用。但不会弹出系统的权限申请框,将被自动批准。

7. Reference

本文是 慌不要慌 原创,发表于 https://danke77.github.io/,请阅读原文支持原创 https://danke77.github.io/2018/06/09/target-android-o/,版权归作者所有,转载请注明出处。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,900评论 25 707
  • 前言 本文主要针对Android O的适配,文中大部分内容将来自官网,本文只是总结提取出适配需要的注意点,关于An...
    GrayMonkey阅读 4,072评论 0 5
  • 前言 谷歌刚刚发布了Android O技术预览版,为了提升续航和减少耗电,Android进一步限制了后台服务的运行...
    客家小罗阅读 7,197评论 1 14
  • Mac环境下安装PhpRedis 首先上项目地址PhpRedis 坑1 自带PHP和MAMP下的PHP 项目地址中...
    YxYYxY阅读 5,691评论 0 4
  • 结尾,开始…… 总有一种悔恨叫自己不努力,总有一种落差叫理想与现实,总有一种孤独叫你怎么样,总有一种失...
    紫荆棘阅读 152评论 0 0