国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Android > 正文

Android APK應用安裝原理解析之AndroidManifest使用PackageParser.parserPackage原理分析

2019-10-22 18:18:47
字體:
來源:轉載
供稿:網友

本文實例講述了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程序設計有所幫助。


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 康乐县| 容城县| 库伦旗| 马公市| 嘉义县| 永丰县| 犍为县| 武川县| 监利县| 大理市| 石景山区| 江陵县| 贵溪市| 宜良县| 信宜市| 弋阳县| 宝坻区| 通河县| 辽宁省| 伊金霍洛旗| 东乡县| 泽州县| 淮滨县| 水城县| 丹凤县| 定襄县| 霍林郭勒市| 福贡县| 克拉玛依市| 铁岭县| 龙海市| 宿松县| 望奎县| 方山县| 神农架林区| 黔江区| 张掖市| 伊宁县| 习水县| 武平县| 左权县|