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

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

概述本文实例讲述了AndroidAPK应用安装之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,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,outError)) {          return null;        }      } else if (tagname.equals("permission-group")) {        if (parsePermissionGroup(pkg,outError) == null) {          return null;        }      } else if (tagname.equals("permission")) {        if (parsePermission(pkg,outError) == null) {          return null;        }      } else if (tagname.equals("permission-tree")) {        if (parsePermissionTree(pkg,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 APK应用安装原理解析之AndroidManifest使用PackageParser.parserPackage原理分析全部内容,希望文章能够帮你解决Android APK应用安装原理解析之AndroidManifest使用PackageParser.parserPackage原理分析所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/web/1143263.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2022-05-31
下一篇2022-05-31

发表评论

登录后才能评论

评论列表(0条)

    保存