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

本文实例讲述了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程序设计有所帮助。

(0)

相关推荐

  • Android无需root实现apk的静默安装

    Android的静默安装似乎是一个很有趣很诱人的东西,但是,用普通做法,如果手机没有root权限的话,似乎很难实现静默安装,因为Android并不提供显示的Intent调用,一般是通过以下方式安装apk: Intent intent = new Intent(Intent.ACTION_VIEW); intent.setDataAndType(Uri.fromFile(file), "application/vnd.android.package-archive"); startAct

  • Android实现检查并下载APK更新、安装APK及获取网络信息的方法

    本文所述实例为一个天气预报中的android代码,主要包括了下载和安装APK.检查Apk更新.显示'已经是最新'或者'无法获取版本信息'对话框.获取当前客户端版本信息.显示版本更新通知对话框.显示下载对话框.判断是否挂载了SD卡.显示文件大小格式:2个小数点显示等.具体实现代码如下: import java.io.ByteArrayInputStream; import java.io.File; import java.io.FileOutputStream; import java.io.I

  • 深入AndroidManifest.xml文件解析详解

    一.关于AndroidManifest.xml AndroidManifest.xml 是每个android程序中必须的文件.它位于整个项目的根目录,描述了package中暴露的组件(activities, services, 等等),他们各自的实现类,各种能被处理的数据和启动位置. 除了能声明程序中的Activities, ContentProviders, Services, 和Intent Receivers,还能指定permissions和instrumentation(安全控制和测试)

  • AndroidManifest.xml配置文件解析

    AndroidManifest.xml配置文件对于Android应用开发来说是非常重要的基础知识,本文旨在总结该配置文件中重点的用法,以便日后查阅.下面是一个标准的AndroidManifest.xml文件样例. 复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?> <manifest> <!-- 基本配置 --> <uses-permission /> <permi

  • 在AndroidManifest.xml中uses-sdk内属性意思

    在AndroidMenifest.xml中,常常会有下面的语句: 复制代码 代码如下: <uses-sdk android:minSdkVersion="4" android:targetSdkVersion="10" android:maxSdkVersion="10" /> 在default.properties中,会看到下面的语句: target=android-10 如果是使用Eclipse的话,还可能会看到这样的警告: At

  • Android学习笔记之AndroidManifest.xml文件解析(详解)

    一.关于AndroidManifest.xml AndroidManifest.xml 是每个android程序中必须的文件.它位于整个项目的根目录,描述了package中暴露的组件(activities, services, 等等),他们各自的实现类,各种能被处理的数据和启动位置. 除了能声明程序中的Activities, ContentProviders, Services, 和Intent Receivers,还能指定permissions和instrumentation(安全控制和测试)

  • Android编程实现监控apk安装,卸载,替换的方法

    本文实例讲述了Android编程实现监控apk安装,卸载,替换的方法.分享给大家供大家参考,具体如下: public class GetBroadcast extends BroadcastReceiver { private static GetBroadcast mReceiver = new GetBroadcast(); private static IntentFilter mIntentFilter; public static void registerReceiver(Conte

  • Android模拟器中安装apk的方法

    本文讲述了Android模拟器中安装apk的方法.分享给大家供大家参考,具体如下: 第一步: 在Eclipse中启动模拟器. 第二步: 打开doc命名窗口,转到你android sdk安装目录中的platform-tools下, 如我的是 E:/Program Files/andriod_sdk/platform-tools 第三步: 利用adb来对android进行安装操作.建议第三步使用方案一 方案一: 为了方便,我们应将要安装的apk直接拷到platform-tools所在的目录下,然后在

  • Android简单判断某个APK是否已经安装的方法

    本文实例讲述了Android简单判断某个APK是否已经安装的方法.分享给大家供大家参考,具体如下: privateboolean isAppInstalled(String uri){ PackageManager pm = getPackageManager(); boolean installed =false; try{ pm.getPackageInfo(uri,PackageManager.GET_ACTIVITIES); installed =true; }catch(Package

  • Android中获取apk安装包信息的方法

    一.获取安装包信息 复制代码 代码如下: /** * 获取apk包的信息:版本号,名称,图标等 * @param absPath apk包的绝对路径 * @param context  */  public void apkInfo(String absPath,Context context) { PackageManager pm = context.getPackageManager();      PackageInfo pkgInfo = pm.getPackageArchiveInf

  • Android 监听apk安装替换卸载广播的实现代码

    首先是要获取应用的安装状态,通过广播的形式以下是和应用程序相关的Broadcast ActionACTION_PACKAGE_ADDED 一个新应用包已经安装在设备上,数据包括包名(最新安装的包程序不能接收到这个广播)ACTION_PACKAGE_REPLACED 一个新版本的应用安装到设备,替换之前已经存在的版本ACTION_PACKAGE_CHANGED 一个已存在的应用程序包已经改变,包括包名ACTION_PACKAGE_REMOVED 一个已存在的应用程序包已经从设备上移除,包括包名(正

随机推荐