项目中遇到了安装APP时,出现“包解析错误”错误,日志中也没有明显的fatal信息,应该是系统报错信息。

Android开发吗,抓日志是必备的技能,没有错误日志,那就抓了全量的logcat日志,开始疯狂的看日志,发现一个错误。

Permission Denial: opening provider android.support.v4.content.FileProvider from ProcessRecord{ca55219 13782:com.android.packageinstaller/u0a32} (pid=13782, uid=10032) that is not exported from UID 10076

哈哈哈,有错误日志,就可以搜了,解决问题近在眼前。日志上面显示没有权限,然后我检查了下代码,网上说,除了要加 Intent.FLAG_GRANT_READ_URI_PERMISSION 还要加上 Intent.FLAG_GRANT_WRITE_URI_PERMISSION ,其实仔细想想,我这只是简单的安装个App,没有写数据啊,为了怕前人留的坑,还是加了这个权限,运行程序,果然是没有用,还是需要继续。

我在想是不是FileProvider的权限问题,检测下FileProvider是否给与权限了,配置如下

<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="${applicationid}.fileprovider"
    android:exported="false"
    android:grantUriPermissions="true">
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/filepaths" />
</provider>

从配置文件看,该给的权限都给了,是不是filepaths指的路径问题,于是检测了下filepaths配置文件,

<external-path
        name="external_storage_root"
        path="." />

简单的说下这个配置文件,每个字段代表的意义。

path:需要临时授权访问的路径(.代表所有路径) name:就是你给这个访问路径起个名字

那么标签external-path是什么意思呢? 代表外部存储,具体的如下:

<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <root-path name="root" path="" /> //代表设备的根目录new File("/");
    <files-path name="files" path="" /> //context.getFilesDir()
    <cache-path name="cache" path="" /> //context.getCacheDir()
    <external-path name="external" path="" /> //Environment.getExternalStorageDirectory()
    <external-files-path name="name" path="path" /> //context.getExternalFilesDirs()
     <external-cache-path name="name" path="path" /> //getExternalCacheDirs()
</paths>

既然了解了,我指定的也是没有毛病啊,但是还是不行,我重新看了下报错信息,发现报错信息中有

that is not exported

是不是在provider中exported的参数设置问题,感觉找到了希望,我急忙去改这个参数,运行程序,噢噢,直接崩溃了。这是什么鬼。看下日志:

 java.lang.SecurityException: Provider must not be exported

大概的意思就是不能设置为exported为true,看来问题还是没有找到。

我重新梳理下,安装APP的适配问题,fileprovider适配其实分三个阶段,

分6.0之前,7.0和7.0之后,也就说在7.0之前根本不需要使用FileProvider, 在6.0之前直接使用

Intent intent = new Intent(Intent.ACTION_VIEW);
Uri.fromFile(new File(installPath)
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);

就可以了,但是到了7.0,Google改变了策略,就需要使用FileProvider,

Intent intent = new Intent();
Uri uri = null;
uri = FileProvider.getUriForFile(context, context.getPackageName()+".fileprovider", new File(filesPath));
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setAction(Intent.ACTION_VIEW);
<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="${APPLICATIONID}.FileProvider"
    android:exported="false"
    android:grantUriPermissions="true"
    tools:replace="android:authorities">
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/file_path" />
</provider>

authorities:app的包名.fileProvider

grantUriPermissions:必须是true,表示授予 URI 临时访问权限

exported:必须是false 否则会报上面的错误

resource:中的@xml/file_paths是我们接下来要添加的文件,具体怎么写,可以参考上面的写法

在8.0及8.0以上的Google又做了改变,我靠。安装APP时,还需要将APP拷贝到context.getFilesDir()这个目录下,但是仍然需要fileprovider,代码如下:

Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addCategory("android.intent.category.DEFAULT");
Uri contentUri = FileProvider.getUriForFile(context, "com.iflytek.appstroe2.fileprovider", new File(dstPath));
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(contentUri, context.getContentResolver().getType(contentUri));
List<ResolveInfo> resInfoList = context.getPackageManager().queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
for (ResolveInfo resolveInfo : resInfoList) {
    context.grantUriPermission(resolveInfo.activityInfo.packageName, contentUri,Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
context.startActivity(intent);

到此算是梳理清楚了,但是我问题还是没有解决。我看了下我的系统版本,发现尼玛是Android8.0的,但是我一直在搞Android7.0的,我按照Android 8.0的方法试了下,发现可以了。

从解决这个的问题过程可以看出,即使一个小问题,在毫无方案的情况下,还是需要回归到问题的本质,这样才能才能有针对性的给出方案。

  • 私信
  •