Magic Studio

专心做有逼格的APP!

android APK应用安装过程以及默认安装路径[转]

一:安装过程

APK是类似Symbian Sis或Sisx的文件格式。通过将APK文件直接传到Android模拟器或Android手机中执行即可安装。

Android应用安装有如下四种方式


1.        系统应用安装――开机时完成,没有安装界面


2.        网络下载应用安装――通过market应用完成,没有安装界面


3.        ADB工具安装――没有安装界面。


4.        第三方应用安装――通过SD卡里的APK文件安装,有安装界面,由packageinstaller.apk应用处理安装及卸载过程的界面。


应用安装的流程及路径 
应用安装涉及到如下几个目录:


system/app 
 系统自带的应用程序,无法删除
 
data/app
 用户程序安装的目录,有删除权限。


安装时把apk文件复制到此目录
 
data/data
 存放应用程序的数据
 
Data/dalvik-cache
 将apk中的dex文件安装到dalvik-cache目录下(dex文件是dalvik虚拟机的可执行文件,其大小约为原始apk文件大小的四分之一)
 


       安装过程:复制APK安装包到data/app目录下,解压并扫描安装包,把dex文件(Dalvik字节码)保存到dalvik-cache目录,并data/data目录下创建对应的应用数据目录。


       卸载过程:删除安装过程中在上述三个目录下创建的文件及目录。




 


一、系统应用安装: 
PackageManagerService处理各种应用的安装,卸载,管理等工作,开机时由systemServer启动此服务


(源文件路径:android\frameworks\base\services\java\com\android\server\PackageManagerService.java)


PackageManagerService服务启动的流程:


1. 首先扫描安装“system\framework”目录下的jar包


1. scanDirLI(mFrameworkDir,PackageParser.PARSE_IS_SYSTEM,


                    scanMode | SCAN_NO_DEX);




2.第二步扫描安装“system\app”目录下的各个系统应用


scanDirLI(mSystemAppDir,PackageParser.PARSE_IS_SYSTEM, scanMode);




3.第三步扫描“data\app”目录,即用户安装的第三方应用


scanDirLI(mAppInstallDir, 0, scanMode);




4.第四步扫描" data\app-private"目录,即安装DRM保护的APK文件(目前没有遇到过此类的应用)。


scanDirLI(mDrmAppPrivateInstallDir,0, scanMode | SCAN_FORWARD_LOCKED);


安装应用的过程


1.scanDirLI(Filedir, int flags, int scanMode) 遍历安装指定目录下的文件


2.scanPackageLI(FilescanFile,


            File destCodeFile, FiledestResourceFile, int parseFlags,


            int scanMode)                安装package文件


3.scanPackageLI(


        File scanFile, File destCodeFile, FiledestResourceFile,


        PackageParser.Package pkg, intparseFlags, int scanMode)


通过解析安装包parsePackage获取到安装包的信息结构


4.mInstaller.install(pkgName,pkg.applicationInfo.uid,


              pkg.applicationInfo.uid);   实现文件复制的安装过程


(源文件路径:frameworks\base\cmds\installd\installd.install)




二、从market上下载应用: 
Google Market应用需要使用gmail账户登录才可以使用,选择某一应用后,开始下载安装包,此过程中,在手机的信号区有进度条提示,下载完成后,会自动调用Packagemanager的接口安装,调用接口如下:


public voidinstallPackage(final Uri packageURI, final IPackageInstallObserver observer,final int flags)


final Uri packageURI:文件下载完成后保存的路径


final IPackageInstallObserver observer:处理返回的安装结果


final int flags:安装的参数,从market上下载的应用,安装参数为-r (replace)


installPackage接口函数的安装过程:


1.public voidinstallPackage(


            final Uri packageURI, final IPackageInstallObserverobserver, final int flags,


            final String installerPackageName)


final StringinstallerPackageName:安装完成后此名称保存在settings里,一般为null,不是关键参数


2.FiletmpPackageFile = copyTempInstallFile(packageURI, res);


把apk文件复制到临时目录下的临时文件


3.private voidinstallPackageLI(Uri pPackageURI,


            int pFlags, boolean newInstall,String installerPackageName,


           File tmpPackageFile, PackageInstalledInfo res)


解析临时文件,获取应用包名pkgName = PackageParser.parsePackageName(


                   tmpPackageFile.getAbsolutePath(), 0);


4.判断如果带有参数INSTALL_REPLACE_EXISTING,则调用replacePackageLI(pkgName,


                        tmpPackageFile,


                        destFilePath,destPackageFile, destResourceFile,


                        pkg, forwardLocked,newInstall, installerPackageName,


                        res)


5.如果没有,则调用installNewPackageLI(pkgName,


                        tmpPackageFile,


                        destFilePath,destPackageFile, destResourceFile,


                        pkg,forwardLocked, newInstall, installerPackageName,


                        res);


6.privatePackageParser.Package scanPackageLI(


        File scanFile, File destCodeFile, FiledestResourceFile,


        PackageParser.Package pkg, intparseFlags, int scanMode)


scanPackageLI以后的流程,与开机时的应用安装流程相同。


三、从ADB工具安装 
Android Debug Bridge (adb) 是SDK自带的管理设备的工具,通过ADB命令行的方式也可以为手机或模拟器安装应用,其入口函数源文件为pm.java


(源文件路径:android\frameworks\base\cmds\pm\src\com\android\commands\pm\pm.java)


ADB命令行的形式为adb install <path_to_apk> ,还可以带安装参数如:"-l""-r" "-i" "-t"


函数runInstall()中判断参数


"-l"――INSTALL_FORWARD_LOCK


 "-r"——INSTALL_REPLACE_EXISTING 


"-i" ——installerPackageName


"-t"——INSTALL_ALLOW_TEST


我们常用的参数为-r,表示覆盖安装手机上已安装的同名应用。从market上下载的应用,也是直接传入这个参数安装的。


runInstall与market调用同样的接口完成应用安装。


public voidinstallPackage(android.net.Uri packageURI,android.content.pm.IPackageInstallObserver observer, int flags,java.lang.String installerPackageName)


四、第三方应用安装――通过SD卡里的APK文件安装 
把APK安装包保存在SD卡中,从手机里访问SD卡中的APK安装包,点击就可以启动安装界面,系统应用Packageinstaller.apk处理这种方式下的安装及卸载界面流程,如下图:




PackageInstallerActivity负责解析包,判断是否是可用的Apk文件


创建临时安装文件/data/data/com.android.packageinstaller/files/ApiDemos.apk


并启动安装确认界面startInstallConfirm,列出解析得到的该应用基本信息。如果手机上已安装有同名应用,则需要用户确认是否要替换安装。


确认安装后,启动InstallAppProgress,调用安装接口完成安装。


pm.installPackage(mPackageURI,observer, installFlags);


其它: 
1. PackageManagerService.java的内部类AppDirObserver实现了监听app目录的功能:当把某个APK拖到app目录下时,可以直接调用scanPackageLI完成安装。


2.手机数据区目录“data/system/packages.xml”文件中,包含了手机上所有已安装应用的基本信息,如安装路径,申请的permission等信息。

 

二:更改默认安装路径

    如果你是一个apk应用开发者,没用android系统全套代码,你只想更改你的APK的默认安装路径的话,可以按下面方法来更改:

 

在AndroidManifest.xml文件Manifest标签中添加android:installLocation属性

android:installLocation

这个属性设置的是默认安装位置, 共有三个有效值,auto、internalOnly、preferExternal

对应表

auto 表示自动,由系统决定安装位置,如果系统手机内存足够,默认安装在手机内存里,如果手机内在不够,则会安装在T卡内

internalOnly 安装在手机内存

preferExternal 安装在外部存储中

Constant Value Description
auto 0 Let the system decide install location
internalOnly 1 Explicitly request to be installed on internal phone storage only
preferExternal 2 Perfer to be installed on SD card, There is no guarantee that the system will honor this request. The application might end up being installed on internal storage if external media is unavailable or too full

 

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="com.ray"
      android:installLocation="auto"
      android:versionCode="1"
      android:versionName="1.0">

如果你是一个手机方案开发者,能改framework层的代码,你希望下载到手机里的APK都默认优先安装到T卡的话,可以按下面方法更改:

在PackageParser.java文件里,将private static final int PARSE_DEFAULT_INSTALL_LOCATION  这个PARSE_DEFAULT_INSTALL_LOCATION值,改为PackageInfo.INSTALL_LOCATION_PREFER_EXTERNAL,就可以了.默认下载就会下到T卡里,T卡不存在或者满了时,才会下到手机内部. 或者如果在应用的AndroidManifest.xml里,指定了installLocation属性为手机内部存储的话,是会存在手机内部的.没指这定的话,PARSE_DEFAULT_INSTALL_LOCATION这个值指定什么,就下到哪里

安装在T卡里的应用,有几个缺点:
1:因为android装载T卡是在系统启动完成后,才开始装载的.所以如果应用装在T卡里的话,那么它是收不到开机启动完成的BOOT_COMPLETED这个广播的
2:如果T卡损坏或者T卡拨出来了,那么装在T卡里的应用是启动不了的.

 

 

系统应用安装的详细过程,老罗有篇文章写得很透彻,这里引用下:

 

Android系统在启动的过程中,会启动一个应用程序管理服务PackageManagerService,这个服务负责扫描系统中特定的目录,找到里面的应用程序文件,即以Apk为后缀的文件,然后对这些文件进解析,得到应用程序的相关信息,完成应用程序的安装过程,本文将详细分析这个过程。

        应用程序管理服务PackageManagerService安装应用程序的过程,其实就是解析析应用程序配置文件AndroidManifest.xml的过程,并从里面得到得到应用程序的相关信息,例如得到应用程序的组件Activity、Service、Broadcast Receiver和Content Provider等信息,有了这些信息后,通过ActivityManagerService这个服务,我们就可以在系统中正常地使用这些应用程序了。

        应用程序管理服务PackageManagerService是系统启动的时候由SystemServer组件启动的,启后它就会执行应用程序安装的过程,因此,本文将从SystemServer启动PackageManagerService服务的过程开始分析系统中的应用程序安装的过程。

        应用程序管理服务PackageManagerService从启动到安装应用程序的过程如下图所示:

        下面我们具体分析每一个步骤。

        Step 1. SystemServer.main

        这个函数定义在frameworks/base/services/java/com/android/server/SystemServer.java文件中:

 

[java] view plaincopy
 
 
  1. public class SystemServer  
  2. {  
  3.     ......  
  4.   
  5.     native public static void init1(String[] args);  
  6.   
  7.     ......  
  8.   
  9.     public static void main(String[] args) {  
  10.         ......  
  11.   
  12.         init1(args);  
  13.   
  14.         ......  
  15.     }  
  16.   
  17.     ......  
  18. }  

        SystemServer组件是由Zygote进程负责启动的,启动的时候就会调用它的main函数,这个函数主要调用了JNI方法init1来做一些系统初始化的工作。

 

        Step 2. SystemServer.init1

        这个函数是一个JNI方法,实现在 frameworks/base/services/jni/com_android_server_SystemServer.cpp文件中:

 

[cpp] view plaincopy
 
 
  1. namespace android {  
  2.   
  3. extern "C" int system_init();  
  4.   
  5. static void android_server_SystemServer_init1(JNIEnv* env, jobject clazz)  
  6. {  
  7.     system_init();  
  8. }  
  9.   
  10. /* 
  11.  * JNI registration. 
  12.  */  
  13. static JNINativeMethod gMethods[] = {  
  14.     /* name, signature, funcPtr */  
  15.     { "init1""([Ljava/lang/String;)V", (void*) android_server_SystemServer_init1 },  
  16. };  
  17.   
  18. int register_android_server_SystemServer(JNIEnv* env)  
  19. {  
  20.     return jniRegisterNativeMethods(env, "com/android/server/SystemServer",  
  21.             gMethods, NELEM(gMethods));  
  22. }  
  23.   
  24. }; // namespace android  

        这个函数很简单,只是调用了system_init函数来进一步执行操作。

 

        Step 3. libsystem_server.system_init

        函数system_init实现在libsystem_server库中,源代码位于frameworks/base/cmds/system_server/library/system_init.cpp文件中:

 

[cpp] view plaincopy
 
 
  1. extern "C" status_t system_init()  
  2. {  
  3.     LOGI("Entered system_init()");  
  4.   
  5.     sp<ProcessState> proc(ProcessState::self());  
  6.   
  7.     sp<IServiceManager> sm = defaultServiceManager();  
  8.     LOGI("ServiceManager: %p\n", sm.get());  
  9.   
  10.     sp<GrimReaper> grim = new GrimReaper();  
  11.     sm->asBinder()->linkToDeath(grim, grim.get(), 0);  
  12.   
  13.     char propBuf[PROPERTY_VALUE_MAX];  
  14.     property_get("system_init.startsurfaceflinger", propBuf, "1");  
  15.     if (strcmp(propBuf, "1") == 0) {  
  16.         // Start the SurfaceFlinger  
  17.         SurfaceFlinger::instantiate();  
  18.     }  
  19.   
  20.     // Start the sensor service  
  21.     SensorService::instantiate();  
  22.   
  23.     // On the simulator, audioflinger et al don't get started the  
  24.     // same way as on the device, and we need to start them here  
  25.     if (!proc->supportsProcesses()) {  
  26.   
  27.         // Start the AudioFlinger  
  28.         AudioFlinger::instantiate();  
  29.   
  30.         // Start the media playback service  
  31.         MediaPlayerService::instantiate();  
  32.   
  33.         // Start the camera service  
  34.         CameraService::instantiate();  
  35.   
  36.         // Start the audio policy service  
  37.         AudioPolicyService::instantiate();  
  38.     }  
  39.   
  40.     // And now start the Android runtime.  We have to do this bit  
  41.     // of nastiness because the Android runtime initialization requires  
  42.     // some of the core system services to already be started.  
  43.     // All other servers should just start the Android runtime at  
  44.     // the beginning of their processes's main(), before calling  
  45.     // the init function.  
  46.     LOGI("System server: starting Android runtime.\n");  
  47.   
  48.     AndroidRuntime* runtime = AndroidRuntime::getRuntime();  
  49.   
  50.     LOGI("System server: starting Android services.\n");  
  51.     runtime->callStatic("com/android/server/SystemServer""init2");  
  52.   
  53.     // If running in our own process, just go into the thread  
  54.     // pool.  Otherwise, call the initialization finished  
  55.     // func to let this process continue its initilization.  
  56.     if (proc->supportsProcesses()) {  
  57.         LOGI("System server: entering thread pool.\n");  
  58.         ProcessState::self()->startThreadPool();  
  59.         IPCThreadState::self()->joinThreadPool();  
  60.         LOGI("System server: exiting thread pool.\n");  
  61.     }  
  62.   
  63.     return NO_ERROR;  
  64. }  

        这个函数首先会初始化SurfaceFlinger、SensorService、AudioFlinger、MediaPlayerService、CameraService和AudioPolicyService这几个服务,然后就通过系统全局唯一的AndroidRuntime实例变量runtime的callStatic来调用SystemServer的init2函数了。关于这个AndroidRuntime实例变量runtime的相关资料,可能参考前面一篇文章Android应用程序进程启动过程的源代码分析一文。

 

        Step 4. AndroidRuntime.callStatic

        这个函数定义在frameworks/base/core/jni/AndroidRuntime.cpp文件中:

 

[cpp] view plaincopy
 
 
  1. /* 
  2. * Call a static Java Programming Language function that takes no arguments and returns void. 
  3. */  
  4. status_t AndroidRuntime::callStatic(const char* className, const char* methodName)  
  5. {  
  6.     JNIEnv* env;  
  7.     jclass clazz;  
  8.     jmethodID methodId;  
  9.   
  10.     env = getJNIEnv();  
  11.     if (env == NULL)  
  12.         return UNKNOWN_ERROR;  
  13.   
  14.     clazz = findClass(env, className);  
  15.     if (clazz == NULL) {  
  16.         LOGE("ERROR: could not find class '%s'\n", className);  
  17.         return UNKNOWN_ERROR;  
  18.     }  
  19.     methodId = env->GetStaticMethodID(clazz, methodName, "()V");  
  20.     if (methodId == NULL) {  
  21.         LOGE("ERROR: could not find method %s.%s\n", className, methodName);  
  22.         return UNKNOWN_ERROR;  
  23.     }  
  24.   
  25.     env->CallStaticVoidMethod(clazz, methodId);  
  26.   
  27.     return NO_ERROR;  
  28. }  

        这个函数调用由参数className指定的java类的静态成员函数,这个静态成员函数是由参数methodName指定的。上面传进来的参数className的值为"com/android/server/SystemServer",而参数methodName的值为"init2",因此,接下来就会调用SystemServer类的init2函数了。

 

        Step 5. SystemServer.init2

        这个函数定义在frameworks/base/services/java/com/android/server/SystemServer.java文件中:

[java] view plaincopy
 
 
  1. public class SystemServer  
  2. {  
  3.     ......  
  4.   
  5.     public static final void init2() {  
  6.         Slog.i(TAG, "Entered the Android system server!");  
  7.         Thread thr = new ServerThread();  
  8.         thr.setName("android.server.ServerThread");  
  9.         thr.start();  
  10.     }  
  11. }  

        这个函数创建了一个ServerThread线程,PackageManagerService服务就是这个线程中启动的了。这里调用了ServerThread实例thr的start函数之后,下面就会执行这个实例的run函数了。

 

        Step 6. ServerThread.run

        这个函数定义在frameworks/base/services/java/com/android/server/SystemServer.java文件中:

[java] view plaincopy
 
 
  1. class ServerThread extends Thread {  
  2.     ......  
  3.   
  4.     @Override  
  5.     public void run() {  
  6.         ......  
  7.   
  8.         IPackageManager pm = null;  
  9.   
  10.         ......  
  11.   
  12.         // Critical services...  
  13.         try {  
  14.             ......  
  15.   
  16.             Slog.i(TAG, "Package Manager");  
  17.             pm = PackageManagerService.main(context,  
  18.                         factoryTest != SystemServer.FACTORY_TEST_OFF);  
  19.   
  20.             ......  
  21.         } catch (RuntimeException e) {  
  22.             Slog.e("System""Failure starting core service", e);  
  23.         }  
  24.   
  25.         ......  
  26.     }  
  27.   
  28.     ......  
  29. }  

        这个函数除了启动PackageManagerService服务之外,还启动了其它很多的服务,例如在前面学习Activity和Service的几篇文章中经常看到的ActivityManagerService服务,有兴趣的读者可以自己研究一下。

 

        Step 7. PackageManagerService.main

        这个函数定义在frameworks/base/services/java/com/android/server/PackageManagerService.java文件中:

 

[cpp] view plaincopy
 
 
  1. class PackageManagerService extends IPackageManager.Stub {  
  2.     ......  
  3.   
  4.     public static final IPackageManager main(Context context, boolean factoryTest) {  
  5.         PackageManagerService m = new PackageManagerService(context, factoryTest);  
  6.         ServiceManager.addService("package", m);  
  7.         return m;  
  8.     }  
  9.   
  10.     ......  
  11. }  

        这个函数创建了一个PackageManagerService服务实例,然后把这个服务添加到ServiceManager中去,ServiceManager是Android系统Binder进程间通信机制的守护进程,负责管理系统中的Binder对象,具体可以参考浅谈Service Manager成为Android进程间通信(IPC)机制Binder守护进程之路一文。
        在创建这个PackageManagerService服务实例时,会在PackageManagerService类的构造函数中开始执行安装应用程序的过程:

 

 

[java] view plaincopy
 
 
  1. class PackageManagerService extends IPackageManager.Stub {  
  2.     ......  
  3.   
  4.     public PackageManagerService(Context context, boolean factoryTest) {  
  5.         ......  
  6.   
  7.         synchronized (mInstallLock) {  
  8.             synchronized (mPackages) {  
  9.                 ......  
  10.   
  11.                 File dataDir = Environment.getDataDirectory();  
  12.                 mAppDataDir = new File(dataDir, "data");  
  13.                 mSecureAppDataDir = new File(dataDir, "secure/data");  
  14.                 mDrmAppPrivateInstallDir = new File(dataDir, "app-private");  
  15.   
  16.                 ......  
  17.   
  18.                 mFrameworkDir = new File(Environment.getRootDirectory(), "framework");  
  19.                 mDalvikCacheDir = new File(dataDir, "dalvik-cache");  
  20.   
  21.                 ......  
  22.   
  23.                 // Find base frameworks (resource packages without code).  
  24.                 mFrameworkInstallObserver = new AppDirObserver(  
  25.                 mFrameworkDir.getPath(), OBSERVER_EVENTS, true);  
  26.                 mFrameworkInstallObserver.startWatching();  
  27.                 scanDirLI(mFrameworkDir, PackageParser.PARSE_IS_SYSTEM  
  28.                     | PackageParser.PARSE_IS_SYSTEM_DIR,  
  29.                     scanMode | SCAN_NO_DEX, 0);  
  30.   
  31.                 // Collect all system packages.  
  32.                 mSystemAppDir = new File(Environment.getRootDirectory(), "app");  
  33.                 mSystemInstallObserver = new AppDirObserver(  
  34.                     mSystemAppDir.getPath(), OBSERVER_EVENTS, true);  
  35.                 mSystemInstallObserver.startWatching();  
  36.                 scanDirLI(mSystemAppDir, PackageParser.PARSE_IS_SYSTEM  
  37.                     | PackageParser.PARSE_IS_SYSTEM_DIR, scanMode, 0);  
  38.   
  39.                 // Collect all vendor packages.  
  40.                 mVendorAppDir = new File("/vendor/app");  
  41.                 mVendorInstallObserver = new AppDirObserver(  
  42.                     mVendorAppDir.getPath(), OBSERVER_EVENTS, true);  
  43.                 mVendorInstallObserver.startWatching();  
  44.                 scanDirLI(mVendorAppDir, PackageParser.PARSE_IS_SYSTEM  
  45.                     | PackageParser.PARSE_IS_SYSTEM_DIR, scanMode, 0);  
  46.   
  47.   
  48.                 mAppInstallObserver = new AppDirObserver(  
  49.                     mAppInstallDir.getPath(), OBSERVER_EVENTS, false);  
  50.                 mAppInstallObserver.startWatching();  
  51.                 scanDirLI(mAppInstallDir, 0, scanMode, 0);  
  52.   
  53.                 mDrmAppInstallObserver = new AppDirObserver(  
  54.                     mDrmAppPrivateInstallDir.getPath(), OBSERVER_EVENTS, false);  
  55.                 mDrmAppInstallObserver.startWatching();  
  56.                 scanDirLI(mDrmAppPrivateInstallDir, PackageParser.PARSE_FORWARD_LOCK,  
  57.                     scanMode, 0);  
  58.   
  59.                 ......  
  60.             }  
  61.         }  
  62.     }  
  63.   
  64.     ......  
  65. }  

        这里会调用scanDirLI函数来扫描移动设备上的下面这五个目录中的Apk文件:

 

        /system/framework

        /system/app

        /vendor/app

        /data/app

        /data/app-private

       Step 8. PackageManagerService.scanDirLI
       这个函数定义在frameworks/base/services/java/com/android/server/PackageManagerService.java文件中:

 

[java] view plaincopy
 
 
  1. class PackageManagerService extends IPackageManager.Stub {  
  2.     ......  
  3.   
  4.     private void scanDirLI(File dir, int flags, int scanMode, long currentTime) {  
  5.         String[] files = dir.list();  
  6.         ......  
  7.   
  8.         int i;  
  9.         for (i=0; i<files.length; i++) {  
  10.             File file = new File(dir, files[i]);  
  11.             if (!isPackageFilename(files[i])) {  
  12.                 // Ignore entries which are not apk's  
  13.                 continue;  
  14.             }  
  15.             PackageParser.Package pkg = scanPackageLI(file,  
  16.                 flags|PackageParser.PARSE_MUST_BE_APK, scanMode, currentTime);  
  17.             // Don't mess around with apps in system partition.  
  18.             if (pkg == null && (flags & PackageParser.PARSE_IS_SYSTEM) == 0 &&  
  19.                 mLastScanError == PackageManager.INSTALL_FAILED_INVALID_APK) {  
  20.                     // Delete the apk  
  21.                     Slog.w(TAG, "Cleaning up failed install of " + file);  
  22.                     file.delete();  
  23.             }  
  24.         }  
  25.     }  
  26.   
  27.   
  28.     ......  
  29. }  

         对于目录中的每一个文件,如果是以后Apk作为后缀名,那么就调用scanPackageLI函数来对它进行解析和安装。

 

         Step 9. PackageManagerService.scanPackageLI

         这个函数定义在frameworks/base/services/java/com/android/server/PackageManagerService.java文件中:

 

[java] view plaincopy
 
 
  1. class PackageManagerService extends IPackageManager.Stub {  
  2.     ......  
  3.   
  4.     private PackageParser.Package scanPackageLI(File scanFile,  
  5.             int parseFlags, int scanMode, long currentTime) {  
  6.         ......  
  7.   
  8.         String scanPath = scanFile.getPath();  
  9.         parseFlags |= mDefParseFlags;  
  10.         PackageParser pp = new PackageParser(scanPath);  
  11.           
  12.         ......  
  13.   
  14.         final PackageParser.Package pkg = pp.parsePackage(scanFile,  
  15.             scanPath, mMetrics, parseFlags);  
  16.   
  17.         ......  
  18.   
  19.         return scanPackageLI(pkg, parseFlags, scanMode | SCAN_UPDATE_SIGNATURE, currentTime);  
  20.     }  
  21.   
  22.     ......  
  23. }  

        这个函数首先会为这个Apk文件创建一个PackageParser实例,接着调用这个实例的parsePackage函数来对这个Apk文件进行解析。这个函数最后还会调用另外一个版本的scanPackageLI函数把来解析后得到的应用程序信息保存在PackageManagerService中。

 

        Step 10. PackageParser.parsePackage
        这个函数定义在frameworks/base/core/java/android/content/pm/PackageParser.java文件中:

 

[java] view plaincopy
 
 
  1. public class PackageParser {  
  2.     ......  
  3.   
  4.     public Package parsePackage(File sourceFile, String destCodePath,  
  5.             DisplayMetrics metrics, int flags) {  
  6.         ......  
  7.   
  8.         mArchiveSourcePath = sourceFile.getPath();  
  9.   
  10.         ......  
  11.   
  12.         XmlResourceParser parser = null;  
  13.         AssetManager assmgr = null;  
  14.         boolean assetError = true;  
  15.         try {  
  16.             assmgr = new AssetManager();  
  17.             int cookie = assmgr.addAssetPath(mArchiveSourcePath);  
  18.             if(cookie != 0) {  
  19.                 parser = assmgr.openXmlResourceParser(cookie, "AndroidManifest.xml");  
  20.                 assetError = false;  
  21.             } else {  
  22.                 ......  
  23.             }  
  24.         } catch (Exception e) {  
  25.             ......  
  26.         }  
  27.   
  28.         ......  
  29.   
  30.         String[] errorText = new String[1];  
  31.         Package pkg = null;  
  32.         Exception errorException = null;  
  33.         try {  
  34.             // XXXX todo: need to figure out correct configuration.  
  35.             Resources res = new Resources(assmgr, metrics, null);  
  36.             pkg = parsePackage(res, parser, flags, errorText);  
  37.         } catch (Exception e) {  
  38.             ......  
  39.         }  
  40.   
  41.         ......  
  42.   
  43.         parser.close();  
  44.         assmgr.close();  
  45.   
  46.         // Set code and resource paths  
  47.         pkg.mPath = destCodePath;  
  48.         pkg.mScanPath = mArchiveSourcePath;  
  49.         //pkg.applicationInfo.sourceDir = destCodePath;  
  50.         //pkg.applicationInfo.publicSourceDir = destRes;  
  51.         pkg.mSignatures = null;  
  52.   
  53.         return pkg;  
  54.     }  
  55.   
  56.     ......  
  57. }  

        每一个Apk文件都是一个归档文件,它里面包含了Android应用程序的配置文件AndroidManifest.xml,这里主要就是要对这个配置文件就行解析了,从Apk归档文件中得到这个配置文件后,就调用另一外版本的parsePackage函数对这个应用程序进行解析了:

 

 

[java] view plaincopy
 
 
  1. public class PackageParser {  
  2.     ......  
  3.   
  4.     private Package parsePackage(  
  5.             Resources res, XmlResourceParser parser, int flags, String[] outError)  
  6.             throws XmlPullParserException, IOException {  
  7.         ......  
  8.   
  9.         String pkgName = parsePackageName(parser, attrs, flags, outError);  
  10.           
  11.         ......  
  12.   
  13.         final Package pkg = new Package(pkgName);  
  14.   
  15.         ......  
  16.   
  17.         int type;  
  18.   
  19.         ......  
  20.           
  21.         TypedArray sa = res.obtainAttributes(attrs,  
  22.             com.android.internal.R.styleable.AndroidManifest);  
  23.   
  24.         ......  
  25.   
  26.         while ((type=parser.next()) != parser.END_DOCUMENT  
  27.             && (type != parser.END_TAG || parser.getDepth() > outerDepth)) {  
  28.                 if (type == parser.END_TAG || type == parser.TEXT) {  
  29.                     continue;  
  30.                 }  
  31.   
  32.                 String tagName = parser.getName();  
  33.                 if (tagName.equals("application")) {  
  34.                     ......  
  35.   
  36.                     if (!parseApplication(pkg, res, parser, attrs, flags, outError)) {  
  37.                         return null;  
  38.                     }  
  39.                 } else if (tagName.equals("permission-group")) {  
  40.                     ......  
  41.                 } else if (tagName.equals("permission")) {  
  42.                     ......  
  43.                 } else if (tagName.equals("permission-tree")) {  
  44.                     ......  
  45.                 } else if (tagName.equals("uses-permission")) {  
  46.                     ......  
  47.                 } else if (tagName.equals("uses-configuration")) {  
  48.                     ......  
  49.                 } else if (tagName.equals("uses-feature")) {  
  50.                     ......  
  51.                 } else if (tagName.equals("uses-sdk")) {  
  52.                     ......  
  53.                 } else if (tagName.equals("supports-screens")) {  
  54.                     ......  
  55.                 } else if (tagName.equals("protected-broadcast")) {  
  56.                     ......  
  57.                 } else if (tagName.equals("instrumentation")) {  
  58.                     ......  
  59.                 } else if (tagName.equals("original-package")) {  
  60.                     ......  
  61.                 } else if (tagName.equals("adopt-permissions")) {  
  62.                     ......  
  63.                 } else if (tagName.equals("uses-gl-texture")) {  
  64.                     ......  
  65.                 } else if (tagName.equals("compatible-screens")) {  
  66.                     ......  
  67.                 } else if (tagName.equals("eat-comment")) {  
  68.                     ......  
  69.                 } else if (RIGID_PARSER) {  
  70.                     ......  
  71.                 } else {  
  72.                     ......  
  73.                 }  
  74.         }  
  75.   
  76.         ......  
  77.   
  78.         return pkg;  
  79.     }  
  80.   
  81.     ......  
  82. }  

        这里就是对AndroidManifest.xml文件中的各个标签进行解析了,各个标签的含义可以参考官方文档http://developer.android.com/guide/topics/manifest/manifest-intro.html,这里我们只简单看一下application标签的解析,这是通过调用parseApplication函数来进行的。

 

        Step 11. PackageParser.parseApplication
        这个函数定义在frameworks/base/core/java/android/content/pm/PackageParser.java文件中:

[java] view plaincopy
 
 
  1. public class PackageParser {  
  2.     ......  
  3.   
  4.     private boolean parseApplication(Package owner, Resources res,  
  5.             XmlPullParser parser, AttributeSet attrs, int flags, String[] outError)  
  6.             throws XmlPullParserException, IOException {  
  7.         final ApplicationInfo ai = owner.applicationInfo;  
  8.         final String pkgName = owner.applicationInfo.packageName;  
  9.   
  10.         TypedArray sa = res.obtainAttributes(attrs,  
  11.             com.android.internal.R.styleable.AndroidManifestApplication);  
  12.   
  13.         ......  
  14.   
  15.         int type;  
  16.         while ((type=parser.next()) != parser.END_DOCUMENT  
  17.             && (type != parser.END_TAG || parser.getDepth() > innerDepth)) {  
  18.                 if (type == parser.END_TAG || type == parser.TEXT) {  
  19.                     continue;  
  20.                 }  
  21.           
  22.                 String tagName = parser.getName();  
  23.                 if (tagName.equals("activity")) {  
  24.                     Activity a = parseActivity(owner, res, parser, attrs, flags, outError, false);  
  25.                     ......  
  26.   
  27.                     owner.activities.add(a);  
  28.   
  29.                 } else if (tagName.equals("receiver")) {  
  30.                     Activity a = parseActivity(owner, res, parser, attrs, flags, outError, true);  
  31.                     ......  
  32.   
  33.                     owner.receivers.add(a);  
  34.                 } else if (tagName.equals("service")) {  
  35.                     Service s = parseService(owner, res, parser, attrs, flags, outError);  
  36.                     ......  
  37.   
  38.                     owner.services.add(s);  
  39.                 } else if (tagName.equals("provider")) {  
  40.                     Provider p = parseProvider(owner, res, parser, attrs, flags, outError);  
  41.                     ......  
  42.   
  43.                     owner.providers.add(p);  
  44.                 } else if (tagName.equals("activity-alias")) {  
  45.                     Activity a = parseActivityAlias(owner, res, parser, attrs, flags, outError);  
  46.                     ......  
  47.   
  48.                     owner.activities.add(a);  
  49.                 } else if (parser.getName().equals("meta-data")) {  
  50.                     ......  
  51.                 } else if (tagName.equals("uses-library")) {  
  52.                     ......  
  53.                 } else if (tagName.equals("uses-package")) {  
  54.                     ......  
  55.                 } else {  
  56.                     ......  
  57.                 }  
  58.         }  
  59.   
  60.         return true;  
  61.     }  
  62.   
  63.     ......  
  64. }  

        这里就是对AndroidManifest.xml文件中的application标签进行解析了,我们常用到的标签就有activity、service、receiver和provider,各个标签的含义可以参考官方文档http://developer.android.com/guide/topics/manifest/manifest-intro.html

 

        这里解析完成后,一层层返回到Step 9中,调用另一个版本的scanPackageLI函数把来解析后得到的应用程序信息保存下来。

        Step 12. PackageManagerService.scanPackageLI

        这个函数定义在frameworks/base/services/java/com/android/server/PackageManagerService.java文件中:

[java] view plaincopy
 
 
  1. class PackageManagerService extends IPackageManager.Stub {  
  2.     ......  
  3.   
  4.     // Keys are String (package name), values are Package.  This also serves  
  5.     // as the lock for the global state.  Methods that must be called with  
  6.     // this lock held have the prefix "LP".  
  7.     final HashMap<String, PackageParser.Package> mPackages =  
  8.         new HashMap<String, PackageParser.Package>();  
  9.   
  10.     ......  
  11.   
  12.     // All available activities, for your resolving pleasure.  
  13.     final ActivityIntentResolver mActivities =  
  14.     new ActivityIntentResolver();  
  15.   
  16.     // All available receivers, for your resolving pleasure.  
  17.     final ActivityIntentResolver mReceivers =  
  18.         new ActivityIntentResolver();  
  19.   
  20.     // All available services, for your resolving pleasure.  
  21.     final ServiceIntentResolver mServices = new ServiceIntentResolver();  
  22.   
  23.     // Keys are String (provider class name), values are Provider.  
  24.     final HashMap<ComponentName, PackageParser.Provider> mProvidersByComponent =  
  25.         new HashMap<ComponentName, PackageParser.Provider>();  
  26.   
  27.     ......  
  28.   
  29.     private PackageParser.Package scanPackageLI(PackageParser.Package pkg,  
  30.             int parseFlags, int scanMode, long currentTime) {  
  31.         ......  
  32.   
  33.         synchronized (mPackages) {  
  34.             ......  
  35.   
  36.             // Add the new setting to mPackages  
  37.             mPackages.put(pkg.applicationInfo.packageName, pkg);  
  38.   
  39.             ......  
  40.   
  41.             int N = pkg.providers.size();  
  42.             int i;  
  43.             for (i=0; i<N; i++) {  
  44.                 PackageParser.Provider p = pkg.providers.get(i);  
  45.                 p.info.processName = fixProcessName(pkg.applicationInfo.processName,  
  46.                     p.info.processName, pkg.applicationInfo.uid);  
  47.                 mProvidersByComponent.put(new ComponentName(p.info.packageName,  
  48.                     p.info.name), p);  
  49.   
  50.                 ......  
  51.             }  
  52.   
  53.             N = pkg.services.size();  
  54.             for (i=0; i<N; i++) {  
  55.                 PackageParser.Service s = pkg.services.get(i);  
  56.                 s.info.processName = fixProcessName(pkg.applicationInfo.processName,  
  57.                     s.info.processName, pkg.applicationInfo.uid);  
  58.                 mServices.addService(s);  
  59.   
  60.                 ......  
  61.             }  
  62.   
  63.             N = pkg.receivers.size();  
  64.             r = null;  
  65.             for (i=0; i<N; i++) {  
  66.                 PackageParser.Activity a = pkg.receivers.get(i);  
  67.                 a.info.processName = fixProcessName(pkg.applicationInfo.processName,  
  68.                     a.info.processName, pkg.applicationInfo.uid);  
  69.                 mReceivers.addActivity(a, "receiver");  
  70.                   
  71.                 ......  
  72.             }  
  73.   
  74.             N = pkg.activities.size();  
  75.             for (i=0; i<N; i++) {  
  76.                 PackageParser.Activity a = pkg.activities.get(i);  
  77.                 a.info.processName = fixProcessName(pkg.applicationInfo.processName,  
  78.                     a.info.processName, pkg.applicationInfo.uid);  
  79.                 mActivities.addActivity(a, "activity");  
  80.                   
  81.                 ......  
  82.             }  
  83.   
  84.             ......  
  85.         }  
  86.   
  87.         ......  
  88.   
  89.         return pkg;  
  90.     }  
  91.   
  92.     ......  
  93. }  

        这个函数主要就是把前面解析应用程序得到的package、provider、service、receiver和activity等信息保存在PackageManagerService服务中了。

 

from:http://blog.csdn.net/xuhui_7810/article/details/11654577

posted on 2014-02-11 14:34  Mr 布鲁斯  阅读(7152)  评论(0编辑  收藏  举报

导航