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下载完成后,可以按如下:
-
通过MD5确认文件下载完整
-
通过反射的办法调用 PackageParser.parseMonolithicPackage(File.class,int.class) 判断APK是否可以安装【重点】
/** * 通过,启动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) 呢,原因是反射获取不到这个类,只能参考该方法反射 PackageParser的 parseMonolithicPackage(sourceFile, 0) 来获取包信息,达到 getPackageInfo(File sourceFile) 一样的效果。
-
再保存新版本相关信息
-
升级界面执行新版本提示
-
安装
以上是个人的一点经验总结,如有理解不对的地方,欢迎各位评论指正,感激不尽。
还没有评论,来说两句吧...