最近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 不可以为:0
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 10
然后就针对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隐藏了
最近Firebase 统计到比较多的Crash就是:android.app.RemoteServiceException: Context.startForegroundService() did not then call Service.startForeground(),这个crash 比较顽强,解决了几次,现在分享下心得。1.堆栈报错信息:2.android.app.RemoteServiceException: Context.startForegroundService() ..
//开启服务做兼容处理
Intent intentOne = new Intent(this, BackService.class);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
startForegroundService(intentOne);
}else {
startServic...
在官方文档Android 8.0 行为变更中有这样一段话:
Android 8.0 有一项复杂功能;系统不允许后台应用创建后台服务。 因此,Android 8.0 引入了一种全新的方法,即Context.startForegroundService(),以在前台启动新服务。
在系统创建服务后,应用有五秒的时间来调用该服务的startForeground()方法以显示新服务的...
Android 8.0 有以下调整:
Android 8.0 的应用尝试在不允许其创建后台服务的情况下使用 startService() 函数,则该函数将引发一个 IllegalStateException。
新的 Context.startForegroundService() 函数将启动一个前台服务。现在,即使应用在后台运行,系统也允许其调用 Context.startForegro...
android.app.RemoteServiceException: Context.startForegroundService() did not then call Service.startForeground()
通过startForegroundService(Intent intent)启动服务时,如果服务没有及时(5s)使用startForeground()响应,那么就会报这个异常。
问题原因和解决
1. Service startForeground之前的代码过于复杂
Your app will crash if you call Context.startForegroundService(…) and then call Context.stopService(…) before Service.startForeground(…) is called.
Some suggestions:
//千年难解的老bug,搜索了一下如下文章分析的不错
stacko...
从 Android 中service 详解 和Android service 启动篇之 startForegroundService 中我们知道在android 8.0 禁止启动后台服务。提出通过startForegroundService() 启动前台服务。但是必须要配合在service 中调用Service.startForeground(),不然就会出现ANR 或者crash。
ANR l...
android.app.RemoteServiceException
Context.startForegroundService() did not then call Service.startForeground()
该异常表示从Package发布通知时存在异常,主要出现在通知栏更新。
[解决方案]:出现这个异常的原因有很多,使用通知需要注意以下几点:
1.每次更新时都需要实例化Notification;
2.设置Notification的ContentView之后记得设置ContentIntent
public class AppService extends Service {
private HashMap<String, Integer> clockMap = new HashMap<String, Integer>();
private String...
项目在部分9.0手机运行报错:startForegroundService() did not then call Service.startForeground()
查资料发现:是因为8.0以上系统不允许后台应用启动后台服务。所以需要把后台服务设置为前台服务。
并且修改service启动函数。
Intent intent = new Intent(getApplicationContex...