Android 7.0/8.0 安装APK时解析包错误问题

在 Android 7.0 引入了“私有目录被限制访问”,通过使用FileProvider来解决问题,但是在安装应用时却出现了解析包出错的问题

 

来检查一下代码

    private void installApk(File file) {
        Intent intent = new Intent(Intent.ACTION_VIEW);
        Uri apkUri;
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            apkUri = FileProvider.getUriForFile(mContext
                    , "项目包名.FileProvider"
                    , file);
        } else {
            apkUri = Uri.fromFile(file);
        }
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        intent.setDataAndType(apkUri, "application/vnd.android.package-archive");
        mContext.startActivity(intent);
    }

因为是在 Android 7.0 才会出现的问题,所以很容易让人想到可能是 FileProvider 造成的,然而检查了半天,还是没找到是什么原因导致的。仔细对比了一下 7.0之前的安装代码,发现了把 intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK) 移到前面,就会正常安装了。

    private void installApk(File file) {
        Intent intent = new Intent(Intent.ACTION_VIEW);
        // 将此段代码移到此,可正常安装
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        Uri apkUri;
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            apkUri = FileProvider.getUriForFile(mContext
                    , "项目包名.FileProvider"
                    , file);
        } else {
            apkUri = Uri.fromFile(file);
        }
        intent.setDataAndType(apkUri, "application/vnd.android.package-archive");
        mContext.startActivity(intent);
    }

 

具体原因

Android 7.0 之后通过 FileProvider 共享文件,因为 exported 为 false,所以需要给目标应用授予临时权限,因此会设置intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);

而在添加此权限之后, intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK) 把添加的权限给替换掉了,因此实际上还是未授予权限。后来将 intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK) 操作放在 addFlags() 之前或替换为intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)即可

 

这里要注意 

setFlags():为intent 设置特殊的标志,会覆盖 intent 已经设置的所有标志。

addFlags():为intent 添加特殊的标志,不会覆盖,只会追加。

 

新增问题

在华为 Android 8.0手机上,遇到了上述方法无法解决的解析包出错问题,记过各种方式检查问题,分析猜测可能在锁屏时用于安装APK的service处于休眠或不可用状态,导致通过 intent.addFlags()方式临时赋予的权限失效。

后改进代码为如下,解决问题

    private void installApk(File file) {
        try{
            Intent intent = new Intent(Intent.ACTION_VIEW);
            intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            Uri apkUri;
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                apkUri = FileProvider.getUriForFile(mContext
                        , "项目包名.FileProvider"
                        , file);
            } else {
                apkUri = Uri.fromFile(file);
            }
            intent.setDataAndType(apkUri, "application/vnd.android.package-archive");
            // 查询所有符合 intent 跳转目标应用类型的应用,注意此方法必须放置在 setDataAndType 方法之后
            List resolveLists = mContext.getPackageManager().queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
            // 然后全部授权
            for (ResolveInfo resolveInfo : resolveLists){
                String packageName = resolveInfo.activityInfo.packageName;
                mContext.grantUriPermission(packageName, apkUri, Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
            }
            mContext.startActivity(intent);
        }catch (Exception e){
            e.printStackTrace();
        }
    }

 

你可能感兴趣的:(下载安装)