Android 7.0 对系统和API做了一些变更。其中关于隐式广播的移除见官方文档链接:Project Svelte:后台优化,根据官方文档介绍,该变更主要是为了帮助优化内存使用和电量消耗,因为隐式广播会在后台频繁启动已注册侦听这些广播的应用,删除这些广播可以显著提升设备性能和用户体验。
针对这一变更,官方文档也给出了应用开发者如何改写应用的一些建议,官方文档参考:后台优化。
面向Android 7.0(API 24)及以上的应用如果在应用的AndroidManifest静态注册CONNECTIVITY_ACTION,将不会收到该广播。但是我们可以使用[Context.registerReceiver()](https://developer.android.com/reference/android/content/Context.html#registerReceiver(android.content.BroadcastReceiver, android.content.IntentFilter))动态注册CONNECTIVITY_ACTION,这将会让我们的应用在运行时收到该广播(这样也可以改善内存使用和电量消耗)。
Android 7.0(API 24)还为我们提供了JobScheduler和onStartJob()(JobService的回调方法),我们可以使用JobInfo.Builder构建一个在Unmetered Network Connection上的JobInfo对象,然后使用JobScheduler执行这个JobInfo对象,接下来就可以在onStartJob()方法里执行我们自己的逻辑。
我们还可以使用ConnectivityManager.NetworkCallback来监听网络变化,比如该接口中的onAvailable()方法。
2) ACTION_NEW_PICTURE或ACTION_NEW_VIDEO
在Android 7.0(API 24)里,这两个隐式广播被移除了,应用程序不能发送或接受这两个广播来监听系统的照片或者视频的拍摄。但是在Android 8.0(API 26),这两个广播又被添加回来了,不过也仅仅可以通过动态注册来接收,并且仅仅适用于监听到他们来执行简单的可以立即被执行的任务,执行复杂的繁重的任务仍然需要使用JobScheduler、JobInfo和onStartJob(),和上面第一点处理方法类似,不同的是,需要调用JobInfo.Builder.addTriggerContentUri()构造JobInfo对象,然后在onStartJob()回调方法里去调用JobParameters.getTriggeredContentAuthorities()和JobParameters.getTriggeredContentUris()来获取拍照或摄像时增加的照片或视频的Uri。
像上面这样优化App可以让App在低内存的设备上改善性能和用户体验(这可以改善Android应用生态环境),移除后台服务(background service)和隐式广播的依赖可以帮助你的App在这样的设备上运行的更好。虽然Android 7.0(API 24)在逐步减少这样的issue,但是我们仍然推荐你通过完全不使用这样的后台进程来优化你的App。
Android 7.0(API 24)还引入了新的adb命令来帮助你测试你的App在不使用后台进程时是否还能很好的运行。命令如下:
//模拟隐式广播和后台服务(background service)不可用的条件
$ adb shell cmd appops set <package_name> RUN_IN_BACKGROUND ignore
//重新使隐式广播和后台服务(background service)可用
$ adb shell cmd appops set <package_name> RUN_IN_BACKGROUND allow