Android APK应用安装原理解析之AndroidManifest使用PackageParser.parserPackage原理分析

作者:zhbinary 时间:2023-05-19 23:58:18 

本文实例讲述了Android APK应用安装之AndroidManifest使用PackageParser.parserPackage原理。分享给大家供大家参考,具体如下:

Android 安装一个APK的时候首先会解析APK,这里要做很多事情,其中一个事情就是解析Manifest.xml文件,并将所有APK的Manifest封装到各种对象中并保存在内存当中

解析Manifest的类是非常重要的,该类就是frameworks\base\core\java\android\content\pm\PackageParser

PackageManagerService会调用PackageParser.parserPackage方法来解析APK清单,下面开始分析PackageParser的实现:

PackageParser是使用的XMLPullParser工具来对XML进行解析的,然后分别通过android.content.pm下各种xxxInfo类来进行封装:

Android APK应用安装原理解析之AndroidManifest使用PackageParser.parserPackage原理分析


public Package parsePackage(File sourceFile, String destCodePath,
 DisplayMetrics metrics, int flags) {
//最后要跑出的解析错误信息
mParseError = PackageManager.INSTALL_SUCCEEDED;
//获得要解析的文件的路径
mArchiveSourcePath = sourceFile.getPath();
//如果要解析的不是文件类型就跳过并且返回该方法
if (!sourceFile.isFile()) {
 Log.w(TAG, "Skipping dir: " + mArchiveSourcePath);
 //更新错误信息
 mParseError = PackageManager.INSTALL_PARSE_FAILED_NOT_APK;
 return null;
}
//如果文件不是以.apk结尾并且flag没有确定一定是APK,那么也返回
if (!isPackageFilename(sourceFile.getName())
   && (flags&PARSE_MUST_BE_APK) != 0) {
 if ((flags&PARSE_IS_SYSTEM) == 0) {
   // We expect to have non-.apk files in the system dir,
   // so don't warn about them.
   Log.w(TAG, "Skipping non-package file: " + mArchiveSourcePath);
 }
 //更新错误信息
 mParseError = PackageManager.INSTALL_PARSE_FAILED_NOT_APK;
 return null;
}
if ((flags&PARSE_CHATTY) != 0 && Config.LOGD) Log.d(
 TAG, "Scanning package: " + mArchiveSourcePath);
XmlResourceParser parser = null;
AssetManager assmgr = null;
boolean assetError = true;
try {
 assmgr = new AssetManager();
 //将一个文件添加到AssetManager中并返回一个唯一标识
 int cookie = assmgr.addAssetPath(mArchiveSourcePath);
 if(cookie != 0) {
   //通过标识去AssetManager中找到标识对应资源中的Manifest清单文件,并返回一个XML的解析器
   parser = assmgr.openXmlResourceParser(cookie, "AndroidManifest.xml");
   //走到这里证明一切顺利
   assetError = false;
 } else {
   Log.w(TAG, "Failed adding asset path:"+mArchiveSourcePath);
 }
} catch (Exception e) {
 Log.w(TAG, "Unable to read AndroidManifest.xml of "
     + mArchiveSourcePath, e);
}
if(assetError) {
 if (assmgr != null) assmgr.close();
 mParseError = PackageManager.INSTALL_PARSE_FAILED_BAD_MANIFEST;
 return null;
}
String[] errorText = new String[1];
Package pkg = null;
Exception errorException = null;
try {
 // XXXX todo: need to figure out correct configuration.
 Resources res = new Resources(assmgr, metrics, null);
 //这个是真正在解析的package的方法,是private method
 pkg = parsePackage(res, parser, flags, errorText);
} catch (Exception e) {
 errorException = e;
 mParseError = PackageManager.INSTALL_PARSE_FAILED_UNEXPECTED_EXCEPTION;
}
if (pkg == null) {
 if (errorException != null) {
   Log.w(TAG, mArchiveSourcePath, errorException);
 } else {
   Log.w(TAG, mArchiveSourcePath + " (at "
       + parser.getPositionDescription()
       + "): " + errorText[0]);
 }
 parser.close();
 assmgr.close();
 if (mParseError == PackageManager.INSTALL_SUCCEEDED) {
   mParseError = PackageManager.INSTALL_PARSE_FAILED_MANIFEST_MALFORMED;
 }
 return null;
}

parserPackage调用了重载的另外一个parserPackage


private Package parsePackage(
   Resources res, XmlResourceParser parser, int flags, String[] outError)
   throws XmlPullParserException, IOException {
   AttributeSet attrs = parser;
   //每次调用这个方法时候清空这些变量
   mParseInstrumentationArgs = null;
   mParseActivityArgs = null;
   mParseServiceArgs = null;
   mParseProviderArgs = null;
   //这里调用这个方法获得包名
   String pkgName = parsePackageName(parser, attrs, flags, outError);
   if (pkgName == null) {
     mParseError = PackageManager.INSTALL_PARSE_FAILED_BAD_PACKAGE_NAME;
     return null;
   }
   int type;
   final Package pkg = new Package(pkgName);
   boolean foundApp = false;
   //从资源里获得AndroidManifest的数组
   TypedArray sa = res.obtainAttributes(attrs,
       com.android.internal.R.styleable.AndroidManifest);
   //继续挖掘出版本号
   pkg.mVersionCode = sa.getInteger(
       com.android.internal.R.styleable.AndroidManifest_versionCode, 0);
   //获取版本名
   pkg.mVersionName = sa.getNonConfigurationString(
       com.android.internal.R.styleable.AndroidManifest_versionName, 0);
   if (pkg.mVersionName != null) {
     pkg.mVersionName = pkg.mVersionName.intern();
   }
   //获得sharedUserId
   String str = sa.getNonConfigurationString(
       com.android.internal.R.styleable.AndroidManifest_sharedUserId, 0);
   if (str != null && str.length() > 0) {
     //验证包名是否符合规则
     String nameError = validateName(str, true);
     if (nameError != null && !"android".equals(pkgName)) {
       outError[0] = "<manifest> specifies bad sharedUserId name \""
         + str + "\": " + nameError;
       mParseError = PackageManager.INSTALL_PARSE_FAILED_BAD_SHARED_USER_ID;
       return null;
     }
     pkg.mSharedUserId = str.intern();
     pkg.mSharedUserLabel = sa.getResourceId(
         com.android.internal.R.styleable.AndroidManifest_sharedUserLabel, 0);
   }
   sa.recycle();
   //安装的位置
   pkg.installLocation = sa.getInteger(
       com.android.internal.R.styleable.AndroidManifest_installLocation,
       PARSE_DEFAULT_INSTALL_LOCATION);
   // Resource boolean are -1, so 1 means we don't know the value.
   int supportsSmallScreens = 1;
   int supportsNormalScreens = 1;
   int supportsLargeScreens = 1;
   int resizeable = 1;
   int anyDensity = 1;
   int outerDepth = parser.getDepth();
   //关键时刻到了,真正的开始解析了
   while ((type=parser.next()) != parser.END_DOCUMENT
       && (type != parser.END_TAG || parser.getDepth() > outerDepth)) {
     if (type == parser.END_TAG || type == parser.TEXT) {
       continue;
     }
     String tagName = parser.getName();
     if (tagName.equals("application")) {
       if (foundApp) {
         if (RIGID_PARSER) {
           outError[0] = "<manifest> has more than one <application>";
           mParseError = PackageManager.INSTALL_PARSE_FAILED_MANIFEST_MALFORMED;
           return null;
         } else {
           Log.w(TAG, "<manifest> has more than one <application>");
           XmlUtils.skipCurrentTag(parser);
           continue;
         }
       }
       foundApp = true;
       if (!parseApplication(pkg, res, parser, attrs, flags, outError)) {
         return null;
       }
     } else if (tagName.equals("permission-group")) {
       if (parsePermissionGroup(pkg, res, parser, attrs, outError) == null) {
         return null;
       }
     } else if (tagName.equals("permission")) {
       if (parsePermission(pkg, res, parser, attrs, outError) == null) {
         return null;
       }
     } else if (tagName.equals("permission-tree")) {
       if (parsePermissionTree(pkg, res, parser, attrs, outError) == null) {
         return null;
       }
     } else if (tagName.equals("uses-permission")) {
       sa = res.obtainAttributes(attrs,
           com.android.internal.R.styleable.AndroidManifestUsesPermission);
       // Note: don't allow this value to be a reference to a resource
       // that may change.
       String name = sa.getNonResourceString(
           com.android.internal.R.styleable.AndroidManifestUsesPermission_name);
       sa.recycle();
      ...................................................
      ...................................................
      ...................................................篇幅有限

这里分别把每种不同的element用不同的小方法去解析,他们的调用顺序是:

Android APK应用安装原理解析之AndroidManifest使用PackageParser.parserPackage原理分析

这些小方法里其实还是有很多小技巧的,有兴趣的话可以细细品位

希望本文所述对大家Android程序设计有所帮助。

来源:http://blog.csdn.net/zhbinary/article/details/7353739

标签:Android,APK,AndroidManifest
0
投稿

猜你喜欢

  • Android编程之自定义锁屏实例分析

    2022-02-02 04:01:50
  • Android编程中的消息机制实例详解

    2022-11-14 06:50:05
  • C#入门教程之ListBox控件使用方法

    2023-09-20 04:52:59
  • spring Roo安装使用简介

    2023-04-10 09:31:50
  • Android蓝牙的开启和搜索设备功能开发实例

    2022-11-07 20:42:17
  • 手动编译C#代码的方法

    2021-05-27 05:42:40
  • Springboot 如何指定获取出 yml文件里面的配置值

    2022-08-29 21:04:48
  • Java求解二叉树的最近公共祖先实例代码

    2023-09-14 18:24:56
  • java队列实现方法(顺序队列,链式队列,循环队列)

    2023-06-24 01:43:17
  • Android中Spinner(下拉框)控件的使用详解

    2021-08-09 01:46:28
  • Java实现的求逆矩阵算法示例

    2023-05-02 03:02:56
  • C#中字符串的一般性和特殊性

    2023-03-23 19:06:07
  • C# Winform调用百度接口实现人脸识别教程(附源码)

    2021-12-27 13:17:09
  • Spring Security内置过滤器的维护方法

    2022-07-30 18:10:16
  • Android实现Back功能代码片段总结

    2021-05-27 01:33:06
  • 浅谈@FeignClient中name和value属性的区别

    2023-11-06 13:04:14
  • Android apk 插件启动内存释放问题

    2022-05-16 07:26:39
  • android中AutoCompleteTextView的简单用法(实现搜索历史)

    2023-01-08 18:33:01
  • Java设计模式之命令模式

    2022-06-17 22:49:07
  • C#使用round函数四舍五入的方法

    2022-01-19 04:33:57
  • asp之家 软件编程 m.aspxhome.com