java.lang.IllegalStateException: Not allowed to start service Intent

先来看下log:

11-05 04:47:32.468   911   911 E AndroidRuntime: FATAL EXCEPTION: main
11-05 04:47:32.468   911   911 E AndroidRuntime: Process: com.shift.phonemanager, PID: 911
11-05 04:47:32.468   911   911 E AndroidRuntime: java.lang.RuntimeException: Unable to start receiver com.shift.phonemanager.apps.accesslog.PermissionAccessStartReceiver: java.lang.IllegalStateException: Not allowed to start service Intent { cmp=com.shift.phonemanager/.apps.accesslog.service.PermissionAccessLogService }: app is in background uid UidRecord{a423648 u0a80 RCVR idle change:uncached procs:1 seq(0,0,0)}
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.app.ActivityThread.handleReceiver(ActivityThread.java:3197)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.app.ActivityThread.-wrap17(Unknown Source:0)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1675)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.os.Handler.dispatchMessage(Handler.java:106)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.os.Looper.loop(Looper.java:164)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.app.ActivityThread.main(ActivityThread.java:6518)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at java.lang.reflect.Method.invoke(Native Method)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:438)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:807)
11-05 04:47:32.468   911   911 E AndroidRuntime: Caused by: java.lang.IllegalStateException: Not allowed to start service Intent { cmp=com.shift.phonemanager/.apps.accesslog.service.PermissionAccessLogService }: app is in background uid UidRecord{a423648 u0a80 RCVR idle change:uncached procs:1 seq(0,0,0)}
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.app.ContextImpl.startServiceCommon(ContextImpl.java:1521)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.app.ContextImpl.startService(ContextImpl.java:1477)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.content.ContextWrapper.startService(ContextWrapper.java:650)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.content.ContextWrapper.startService(ContextWrapper.java:650)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at com.shift.phonemanager.apps.accesslog.PermissionAccessStartReceiver.startAccessLogService(PermissionAccessStartReceiver.java:22)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at com.shift.phonemanager.apps.accesslog.PermissionAccessStartReceiver.onReceive(PermissionAccessStartReceiver.java:18)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	at android.app.ActivityThread.handleReceiver(ActivityThread.java:3190)
11-05 04:47:32.468   911   911 E AndroidRuntime: 	... 8 more

该Exception 是从ContextImpl.java 中抛出来的,详细可以看 Android service 启动篇之 startService

 

代码如下:

    private ComponentName startServiceCommon(Intent service, boolean requireForeground,
            UserHandle user) {
        try {
            validateServiceIntent(service);
            service.prepareToLeaveProcess(this);
            ComponentName cn = ActivityManager.getService().startService(
                mMainThread.getApplicationThread(), service, service.resolveTypeIfNeeded(
                            getContentResolver()), requireForeground,
                            getOpPackageName(), user.getIdentifier());
            if (cn != null) {
                if (cn.getPackageName().equals("!")) {
                    throw new SecurityException(
                            "Not allowed to start service " + service
                            + " without permission " + cn.getClassName());
                } else if (cn.getPackageName().equals("!!")) {
                    throw new SecurityException(
                            "Unable to start service " + service
                            + ": " + cn.getClassName());
                } else if (cn.getPackageName().equals("?")) {
                    throw new IllegalStateException(
                            "Not allowed to start service " + service + ": " + cn.getClassName());
                }
            }
            return cn;
        } catch (RemoteException e) {
            throw e.rethrowFromSystemServer();
        }
    }

当返回的ComponentName 的package name 为 "?" 的时候就会抛出这个异常。

 

通过Android service 启动篇之 startService 的 3.3.1 中我们可以知道,必须要满足一定的条件,应用才能启动后台服务。

 

结论:

顺利启动service,需要满足下面的条件:

  • app 为persistent
  • 或 service 的uid 在background 的白名单中
  • 或 service 的uid 在device id 的白名单中 
  • 对于 service 的应用SDK 版本小于O(26),而且AppOpsManager 中是allowed 状态
  • 对于SDK 大于等于O(26)的service,不满足上面条件只能选择前台服务,通过startForegroundService 启动

 

详细 service 的机制可以看 Android 中service 详解

 

 

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

私房菜

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值