android APK安装提示“解析错误,解析软件包时出现问题” 解析

android APK安装提示“解析错误,解析软件包时出现问题” 解析及优化

1.问题

项目产品在客户端升级的过程中遇到弹窗提示“解析错误,解析软件包时出现问题”,如下图。

在这里插入图片描述

2.分析过程

1.该弹窗内容,搜索项目工程代码,并无相关字眼,排除项目代码中的业务。

2.通过APK安装调用代码,可知安装行为,实为系统应用PackageInstaller中的业务

/** * 安装APK * @param file */
public static void installAPK(Context context,File file) {   
    try {      
        	Intent intent = new Intent();      			                                             intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);            				               intent.setAction("android.intent.action.VIEW");                                           intent.addCategory("android.intent.category.DEFAULT");                                   intent.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive");      
            context.startActivity(intent);   
    } catch (Exception e) 
    {     
        e.printStackTrace();   
    }
}

Packageinstaller清单文件内容,可知上述调用方式实则调用Packageinstaller的PackageInstallerActivity的第一个红色框的隐式调用方法

在这里插入图片描述

4.查询解析错误的相关业务

通过查找res/values-zh-rCN的strings.xml文件中“解析软件包时出现问题”文本,确有该内容

在这里插入图片描述

通过Parse_error_dlg_text字符串资源,反查生成对话框代码位置

在这里插入图片描述

通过DLG_PACKAGE_ERROR查找调用位置,业务调用

showDialogInner(DLG_PACKAGE_ERROR)

显示解析失败对话框,其中有2处调用该方法。

在这里插入图片描述

从上图 onCreat() 方法可看出,调用安装程序的Intent携带的scheme内容为“package”,获取APK包信息=null,则调用第一处的解析错误弹窗;否则执行else代码块,获取apk包信息为null,则调用解析错误弹窗。

从项目代码调用安装程序可知,采用android.intent.action.VIEW隐式调用,从packageinstaller清单文件AndroidManifest.xml可知该方式携带scheme=“file”,所以实际是执行第二处的解析错误弹窗。

else代码块中,通过PackageUtil工具类中的 getPackageInfo(sourceFile) 方法获取apk文件包信息,如返回null,则弹出解析错误对话框。

PackageUtil中的 getPackageInfo(sourceFile) 如下,实际执行是调用 PackageParser类中的parseMonolithicPackage(sourceFile, 0) 方法获取安装包的信息。

    /**
     * Utility method to get package information for a given {@link File}
     */
    public static PackageParser.Package getPackageInfo(File sourceFile) {
        final PackageParser parser = new PackageParser();
        try {
            PackageParser.Package pkg = parser.parseMonolithicPackage(sourceFile, 0);
            parser.collectManifestDigest(pkg);
            return pkg;
        } catch (PackageParserException e) {
            return null;
        }
    }

3.总结

针对安装APK解析错误的问题,经以上分析可知,Packageinstaller调用PackageUtil提取APK的信息,如获取不到,就弹窗解析错误,退出安装流程。

4.优化

为了避免出现解析错误的问题,我们可以在APK下载完成后,可以按如下:

  1. 通过MD5确认文件下载完整

  2. 通过反射的办法调用 PackageParser.parseMonolithicPackage(File.class,int.class) 判断APK是否可以安装【重点】

        /**
         * 通过<data android:scheme="file" />,启动PackageInstaller()
         * @param apkPath
         * @return
         */
        private static boolean isApkOk2(String apkPath){
            try {
    
                Class<?> aClass = Class.forName("android.content.pm.PackageParser");
                Method parseMonolithicPackage = aClass.getMethod("parseMonolithicPackage", File.class,int.class);
                parseMonolithicPackage.setAccessible(true);
                Object invoke = parseMonolithicPackage.invoke(aClass.newInstance(), new File(apkPath), 0);
                if (invoke==null){
    //                Log.w(TAG, "Parse error when parsing manifest. Discontinuing installation");
                    LogUtils.e(TAG,"get APK info is NULL");
                    return false;
                }else{
                    LogUtils.e(TAG,"get APK info is SUCCESS");
                    return true;
                }
            } catch (ClassNotFoundException e) {
                e.printStackTrace();
                return false;
            } catch (NoSuchMethodException e) {
                e.printStackTrace();
                return false;
            } catch (IllegalAccessException e) {
                e.printStackTrace();
                return false;
            } catch (java.lang.InstantiationException e) {
                e.printStackTrace();
                return false;
            } catch (InvocationTargetException e) {
                e.printStackTrace();
                return false;
            }
    
        }
    

至于为什么不反射 PackageUtil 中的 getPackageInfo(File sourceFile) 呢,原因是反射获取不到这个类,只能参考该方法反射 PackageParserparseMonolithicPackage(sourceFile, 0) 来获取包信息,达到 getPackageInfo(File sourceFile) 一样的效果。

  1. 再保存新版本相关信息

  2. 升级界面执行新版本提示

  3. 安装

以上是个人的一点经验总结,如有理解不对的地方,欢迎各位评论指正,感激不尽。

  • 23
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值