赞
踩
项目产品在客户端升级的过程中遇到弹窗提示“解析错误,解析软件包时出现问题”,如下图。
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;
}
}
针对安装APK解析错误的问题,经以上分析可知,Packageinstaller调用PackageUtil提取APK的信息,如获取不到,就弹窗解析错误,退出安装流程。
为了避免出现解析错误的问题,我们可以在APK下载完成后,可以按如下:
通过MD5确认文件下载完整
通过反射的办法调用 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) 呢,原因是反射获取不到这个类,只能参考该方法反射 PackageParser的 parseMonolithicPackage(sourceFile, 0) 来获取包信息,达到 getPackageInfo(File sourceFile) 一样的效果。
再保存新版本相关信息
升级界面执行新版本提示
安装
以上是个人的一点经验总结,如有理解不对的地方,欢迎各位评论指正,感激不尽。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。