Context.startForegroundService() did not then call Service.startForeground()?

最近Firebase 统计到比较多的Crash就是: android.app.RemoteServiceException: Context.startForegroundService() did not then call Service.startForeground(),这个crash 比较顽强,解决了几次,现在分享下心得。

1.堆栈报错信息:

 

2.android.app.RemoteServiceException: Context.startForegroundService() did not then call Service.startForeground() 这个报错原因很好理解就是:

Android 8.0 有一项复杂功能;系统不允许后台应用创建后台服务。 因此,Android 8.0 引入了一种全新的方法,即 Context.startForegroundService(),以在前台启动新服务。

在系统创建服务后,应用有5秒的时间来调用该服务的 startForeground() 方法以显示新服务的用户可见通知。如果应用在此时间限制内未调用 startForeground(),则系统将停止服务并声明此应用为 ANR。

但是目前在调用:context.startForegroundService(intent)时报如下ANR,startForegroundService()文档说明在service启动后要调用startForeground()。

3.知道原因了,正常情况咱们就在startForegroundService后调下startForeground就可以了,理想很丰满,现实很残酷,发现有时即使调用了startForeground也不行,遇到这样的事情就头大了,接下来就是真正解决这个问题了

4.首先注意的是:startForeground的id 不可以为:0Context.startForegroundService() did not then call Service.startForeground()?_第1张图片

 5.有时候发现调用了startForegroundService后 5s内没有走startForeground 网上有人说那就在调用startForegroundService的后自己加个Timer,当调用startForegroundService后的4.5s取消该service: stopService(),这种方法测试了下也是不work的 即使调用了stopService(),依然会crash 

6.有人说既然是crash ,那就调用大招:try{}catch() ,但是在这里try{}catch()也不好使

7.既然发现startForegroundService后,Service 启动起来,会走自己的生命周期,万般无奈就在Service启动的回调方法都调用下startForeground()  对用的回调方法有:Service的onCreate(),onStart(),onStartCommand()

8.按照上面的方法,基本没有问题了,今天早晨看firebase 发现还是有crash,但是这次crash的device都是Android 10Context.startForegroundService() did not then call Service.startForeground()?_第2张图片

 然后就针对Android Q查这个问题,查了很久,竟然说是需要悬浮框的permission,找到问题就好说了,判断是Android10的手机,判断下悬浮框的权限:Settings.canDrawOverlays(context) 如果给了权限就调用:startForegroundService   else 就去要悬浮框的permission:

Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:" + getPackageName()));

startActivityForResult(intent, AppConstant.ActivityRequestCode.OVERLAY_PERMISSION_REQ_CODE);

用户返回后在:onActivityResult判断是否给与了悬浮框的权限,给了再去调用:startForegroundService()

@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if (requestCode == AppConstant.ActivityRequestCode.OVERLAY_PERMISSION_REQ_CODE) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            if (Settings.canDrawOverlays(VoiceRoomActivity.this)) {
               //调用startForegroundService()
            }
        }
    }
}

 9.最后总结下:Android O 后台应用想启动服务调用:startForegroundService(),调用startForegroundService()后 切记调用startForeground(),这个时候会有一个Notification常驻,这个表示APP占用着资源,杀不杀是用户决定,我们也别妄想把这个Notification隐藏了

你可能感兴趣的:(开发技术积累)