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

所属分类: 软件编程 / Android 阅读数: 35
收藏 0 赞 0 分享

本文实例讲述了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类来进行封装:

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相关内容感兴趣的读者可查看本站专题:《Android开发入门与进阶教程》、《Android调试技巧与常见问题解决方法汇总》、《Android基本组件用法总结》、《Android视图View技巧总结》、《Android布局layout技巧总结》及《Android控件用法总结

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

更多精彩内容其他人还在看

Android中加入名片扫描功能实例代码

这篇文章主要介绍了Android中加入名片扫描功能实例代码的相关资料,需要的朋友可以参考下
收藏 0 赞 0 分享

Android仿微信发表说说实现拍照、多图上传功能

这篇文章主要为大家详细介绍了Android仿微信发表说说实现拍照、多图上传功能,使用Retrofit2.0技术,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

设置Android系统永不锁屏永不休眠的方法

在进行Android系统开发的时候,有些特定的情况需要设置系统永不锁屏,永不休眠。本篇文章给大家介绍Android 永不锁屏,开机不锁屏,删除设置中休眠时间选项,需要的朋友一起学习吧
收藏 0 赞 0 分享

Android Retrofit 2.0框架上传图片解决方案

这篇文章主要介绍了Android Retrofit 2.0框架上传一张与多张图片解决方案,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Android自定义等待对话框

这篇文章主要为大家详细介绍了Android自定义等待对话框的实现方法,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Android中Window添加View的底层原理

这篇文章主要介绍了Android中Window添加View的底层原理,需要的朋友可以参考下
收藏 0 赞 0 分享

Android调用系统默认浏览器访问的方法

这篇文章主要介绍了Android调用系统默认浏览器访问的方法的相关资料,需要的朋友可以参考下
收藏 0 赞 0 分享

Android开发退出程序的方法汇总

Android程序有很多Activity,比如说主窗口A,调用了子窗口B,子窗口B又调用子窗口C,back返回子窗口B后,在B中如何关闭整个Android应用程序呢? 下面脚本之家小编就给大家介绍android开发退出程序的几种方法,感兴趣的朋友参考下吧
收藏 0 赞 0 分享

Android程序开发中单选按钮(RadioGroup)的使用详解

在android程序开发中,无论是单选按钮还是多选按钮都非常的常见,接下来通过本文给大家介绍Android程序开发中单选按钮(RadioGroup)的使用,需要的朋友参考下吧
收藏 0 赞 0 分享

Android实现仿网易今日头条等自定义频道listview 或者grideview等item上移到另一个view中

这篇文章主要介绍了Android实现仿网易今日头条等自定义频道listview 或者grideview等item上移到另一个view中 的相关资料,需要的朋友可以参考下
收藏 0 赞 0 分享
查看更多