包可见性引起的问题:Unable to start service Intent U=0: not found

Quibbler 2021-6-18 5697

包可见性引起的问题:

Unable to start service Intent U=0: not found


        有同学遇到无法启动三方应用Service的“玄学”问题,让我来看看到底是哪里的问题。通常启动三方应用的Service很简单,只需简短的几行代码:

    val intent = Intent("cn.quibbler.server.aidl")
    intent.addCategory(Intent.CATEGORY_DEFAULT)
    intent.setPackage("cn.quibbler.myapplication")
    startService(intent)

        但是无论如何也不能启动三方应用的Service服务,查看AMS相关的系统日志发现是找不到Service导致启动失败(也没有报异常出来)




1、问题分析

        上面这种情况下系统的日志和Service没有在AndroidManifest.xml中注册一模一样,查阅很多资料都提到这一点。但是相信四大组件没在AndroidManifest.xml中注册这么“低级”的错误开发者不会犯,并且确定enabledexported已经开启。

    <service
        android:name=".QuibblerService"
        android:enabled="true"
        android:exported="true">
        
        <intent-filter>
            <action android:name="cn.quibbler.server.aidl" />
            <category android:name="android.intent.category.DEFAULT" />
        </intent-filter>
        
    </service>

        无法启动三方应用的Service,但是三方应用内部却可以启动自己的Service。说明Service注册是没问题的。那一定是调用方应用自身的问题。那么是什么问题呢?


        其实是Android 11上Google收缩了程序包可见性,这就导致可能查询不到三方应用及其内部的ContentProviderService组件。Android开发者官网给出的解释如下:

        "如果您的应用以 Android 11(API 级别 30)或更高版本为目标平台,在默认情况下,系统会自动让部分应用对您的应用可见,但会隐藏其他应用。通过让部分应用在默认情况下不可见,系统可以了解应向您的应用显示哪些其他应用,这样有助于鼓励最小权限原则,还可帮助 Google Play 等应用商店评估应用为用户提供的隐私权和安全性。"



2、解决

        这个问题其实就是Android R的适配,有以下几种解决方法:


2.1、降低版本兼容处理(不推荐)

        首先想到的办法就是,降低targetSdkVersion版本,这样也行,运行在Android 11设备上没问题,不过迟早还是需要适配Android这个特性的。

    targetSdkVersion 29


2.2、增加QUERY_ALL_PACKAGES权限(不建议)

        记得曾在Android 11适配的过程中用到新增的一个权限:android.permission.QUERY_ALL_PACKAGES。加上这个权限就能够查询所有的应用,解决前面无法启动三方应用Service的问题。而且QUERY_ALL_PACKAGES权限等级为normal,只要在AndroidManifest.xml中申请声明即可获得该权限。

    <uses-permission android:name="android.permission.QUERY_ALL_PACKAGES" />

        不过,Google限制该权限的使用,如果开发者的应用需要上架Google Play,最好不要使用该权限否则会被检测警告甚至不让上架。



2.3、使用<queries>标签声明使用的三方组件(推荐)

        可以使用范围更具体的的查询,使用<queries>标签声明要查询的<intent>,系统会查询所有能够处理该IntentService等组件。

    <queries>
        <intent>
            <action android:name="cn.quibbler.server.aidl" />
            <data android:path="quibbler" />
        </intent>
    </queries>

        需要注意,开发者不要在<queries>中声明“宽泛的intent”。


        对于ContentProvider,通过<queries>中的<provider>子标签查询指定的authorities

    <queries>
        <provider android:authorities="sadas" />
    </queries>


        还直接添加合作的三方应用包名<package />,使对方包对自己可见。

    <queries>
        <package android:name="com.tencent.qq" />
    </queries>

         开发者应当遵循最小使用原则,只添加应用所需的三方应用包,Google增加这样的设计也是为了隐私和安全考虑。


        最后,并非所有的问题都是玄学问题,如果经常遇到“玄学”反倒是应该好好思考,是不是技术知识掌握太窄。



相关资料:

        Android 11 中的软件包可见性

        管理软件包可见性

        permission > QUERY_ALL_PACKAGES

        

不忘初心的阿甘
最新回复 (0)
    • 安卓笔记本
      2
        登录 注册 QQ
返回
仅供学习交流,切勿用于商业用途。如有错误欢迎指出:fluent0418@gmail.com