以下是我在a blog post 中概述的一些选项:
解决方法 #1:startForegroundService()
您的BroadcastReceiver 接收ACTION_BOOT_COMPLETED 广播
在 Android 上可以调用 startForegroundService() 而不是 startService()
8.0+:
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.Build;
public class OnBootReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Intent i=new Intent(context, TestIntentService.class);
if (Build.VERSION.SDK_INT>=Build.VERSION_CODES.O) {
context.startForegroundService(i);
}
else {
context.startService(i);
}
}
}
请注意,这在一定程度上是有效的,即使您的服务实际上并没有
曾经打电话给startForeground()。你有一个时间来解决
调用startForeground(),“与执行此操作的 ANR 间隔相当”。
如果你的工作时间超过一毫秒但不到几秒,
您可以跳过 Notification 和 startForeground() 通话。然而,
你会在 LogCat 中得到一个错误:
E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.commonsware.myapplication, PID: 5991
android.app.RemoteServiceException: Context.startForegroundService() did not then call Service.startForeground()
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1775)
at android.os.Handler.dispatchMessage(Handler.java:105)
at android.os.Looper.loop(Looper.java:164)
at android.app.ActivityThread.main(ActivityThread.java:6541)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767)
当然,如果您不介意简短地使用Notification,欢迎您
像 Android 期望的那样使用startForeground(),在这种情况下,您可以
后台正常工作,尽管用户通知中显示了一个条目
阴影。
解决方法 #2:goAsync()
BroadcastReceiver 从 API 级别 11 开始提供 goAsync()。这允许您的
接收器完成主应用程序线程的工作,因此您可以摆脱
完全IntentService 并将您的代码移动到BroadcastReceiver。
你仍然只有 ANR
可以使用的超时时间,但您不会占用您的主应用程序
线。这比第一种解决方法要好,因为它具有相同的
时间限制,但避免了令人讨厌的错误。但是,它确实需要一些数量
返工。
解决方法 #3:JobScheduler
如果您的工作将花费超过几秒钟的时间并且您希望避免
Notification,您可以修改代码以实现 JobService 和
与JobScheduler 合作。这有一个额外的好处是只给你
控制何时满足其他标准(例如,有可用的互联网
联系)。但是,这不仅需要重写,而且JobScheduler
仅适用于 Android 5.0+,因此如果您的 minSdkVersion 小于 21,
您将需要在旧设备上使用其他解决方案。
更新:Eugen Pechanec pointed out JobIntentService,
这是一个有趣的JobService/IntentService mashup。