亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

apk的安裝分析

2019-11-09 16:25:42
字體:
來源:轉載
供稿:網友

一、概述

        APK為AndroidPackage的縮寫,Android應用安裝有如下四種方式:

(1)系統應用安裝――開機時完成,沒有安裝界面

(2)網絡下載應用安裝――通過market應用完成,沒有安裝界面

(3)ADB工具安裝――沒有安裝界面

(4)第三方應用安裝――通過SD卡里的APK文件安裝,有安裝界面,由 packageinstaller.apk 應用處理安裝及卸載。

       應用安裝涉及到如下幾個目錄: 

(1)system/app:系統自帶的應用程序,獲得adb root權限才能刪除

(2)data/app:用戶程序安裝的目錄。用戶 安裝時把apk文件 復制 到此目錄

(3)data/data:存放應用程序運行產生的數據

(4)data/dalvik-cache:將apk中的dex文件安裝到dalvik-cache目錄下(dex文件是dalvik虛擬機的可執行文件,其大小約為原始apk文件大小的四分之一)

       安裝過程:

(1)復制APK安裝包到data/app目錄下,解壓并掃描安裝包,

(2)將dex文件(Dalvik字節碼)保存到dalvik-cache目錄,

(3)在data/data目錄下創建對應的應用數據目錄。

       卸載過程:

刪除 安裝過程中在上述三個目錄下創建的文件及目錄。

1、開機安裝 

          PackageManagerService處理各種應用的安裝,卸載,管理等工作,開機時由systemServer啟動此服務

(1)首先掃描安裝“system/framework”目錄下的jar包

scanDirLI(mFrameworkDir, PackageParser.PARSE_IS_SYSTEM, scanMode | SCAN_NO_DEX);

(2)掃描安裝“system/app”目錄下的各個系統應用

scanDirLI(mSystemAppDir,PackageParser.PARSE_IS_SYSTEM, scanMode);

掃描“data/app”目錄,即用戶安裝的第三方應用

scanDirLI(mAppInstallDir, 0, scanMode);

掃描" data/app-PRivate"目錄,即安裝DRM保護的APK文件(目前沒有遇到過此類的應用)

scanDirLI(mDrmAppPrivateInstallDir,0, scanMode | SCAN_FORWARD_LOCKED)

(3)安裝應用的過程

scanDirLI(Filedir, int flags, int scanMode) 遍歷安裝指定目錄下的文件

scanPackageLI(FilescanFile,File destCodeFile, FiledestResourceFile, int parseFlags,int scanMode) 安裝package文件

scanPackageLI(File scanFile, File destCodeFile, FiledestResourceFile,PackageParser.Package pkg, intparseFlags, int scanMode)通過解析安裝包parsePackage獲取到安裝包的信息結構

mInstaller.install(pkgName,pkg.applicationInfo.uid,pkg.applicationInfo.uid); 實現文件復制的安裝過程

2、從market上下載應用

  Google Market應用需要使用Gmail賬戶登錄才可以使用,選擇某一應用后,開始下載安裝包,此過程中,在手機的信號區有進度條提示,下載完成后,會自動調用PackageManager#installPackage接口

3、從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 void installPackage(android.net.Uri packageURI,android.content.pm.ipackageInstallObserver observer, int flags,java.lang.String installerPackageName)

4、第三方應用安裝――通過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);

【PS】

(1)PackageManagerService.java的內部類AppDirObserver實現了監聽app目錄的功能:當把某個APK拖到app目錄下時,可以直接調用scanPackageLI完成安裝。

(2)手機數據區目錄“data/system/packages.xml”文件中,包含了手機上所有已安裝應用的基本信息,如安裝路徑,申請的permission等信息。

二、PMS#installPackage

installPackage方法中主要調用了installPackageAsUser,其源碼如下:

    public void installPackage(String originPath, IPackageInstallObserver2 observer,            int installFlags, String installerPackageName, VerificationParams verificationParams,            String packageAbiOverride) {        installPackageAsUser(originPath, observer, installFlags, installerPackageName, verificationParams,                packageAbiOverride, UserHandle.getCallingUserId());    } 再看installPackageAsUser,

public void installPackageAsUser(String originPath, IPackageInstallObserver2 observer,            int installFlags, String installerPackageName, VerificationParams verificationParams,            String packageAbiOverride, int userId) {        mContext.enforceCallingOrSelfPermission(android.Manifest.permission.INSTALL_PACKAGES, null);        final int callingUid = Binder.getCallingUid();        enforceCrossUserPermission(callingUid, userId, true, true, "installPackageAsUser");        if (isUserRestricted(userId, UserManager.DISALLOW_INSTALL_APPS)) {            try {                if (observer != null) {                    observer.onPackageInstalled("", INSTALL_FAILED_USER_RESTRICTED, null, null);                }            } catch (RemoteException re) {            }            return;        }        if ((callingUid == Process.SHELL_UID) || (callingUid == Process.ROOT_UID)) {            installFlags |= PackageManager.INSTALL_FROM_ADB;        } else {            // Caller holds INSTALL_PACKAGES permission, so we're less strict            // about installerPackageName.            installFlags &= ~PackageManager.INSTALL_FROM_ADB;            installFlags &= ~PackageManager.INSTALL_ALL_USERS;        }        UserHandle user;        if ((installFlags & PackageManager.INSTALL_ALL_USERS) != 0) {            user = UserHandle.ALL;        } else {            user = new UserHandle(userId);        }        // Only system components can circumvent runtime permissions when installing.        if ((installFlags & PackageManager.INSTALL_GRANT_RUNTIME_PERMISSIONS) != 0                && mContext.checkCallingOrSelfPermission(Manifest.permission                .INSTALL_GRANT_RUNTIME_PERMISSIONS) == PackageManager.PERMISSION_DENIED) {            throw new SecurityException("You need the "                    + "android.permission.INSTALL_GRANT_RUNTIME_PERMISSIONS permission "                    + "to use the PackageManager.INSTALL_GRANT_RUNTIME_PERMISSIONS flag");        }        verificationParams.setInstallerUid(callingUid);        final File originFile = new File(originPath);        final OriginInfo origin = OriginInfo.fromUntrustedFile(originFile);        // 直接交給mHandler來處理了,這個mHandler在PackageManagerService構造函數中初始化,是一個PackageHandler實例        final Message msg = mHandler.obtainMessage(INIT_COPY);        msg.obj = new InstallParams(origin, null, observer, installFlags, installerPackageName,                null, verificationParams, user, packageAbiOverride, null);        mHandler.sendMessage(msg);    } 再看PackageHandler類

    class PackageHandler extends Handler {        PackageHandler(Looper looper) {            super(looper);        }        public void handleMessage(Message msg) {            try {                doHandleMessage(msg);            } finally {                Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);            }        }                void doHandleMessage(Message msg) {            switch (msg.what) {                case INIT_COPY: {                    HandlerParams params = (HandlerParams) msg.obj;                    int idx = mPendingInstalls.size();                    if (DEBUG_INSTALL) Slog.i(TAG, "init_copy idx=" + idx + ": " + params);                    // If a bind was already initiated we dont really                    // need to do anything. The pending install                    // will be processed later on.                    if (!mBound) {                        // If this is the only one pending we might                        // have to bind to the service again.                        if (!connectToService()) {                            Slog.e(TAG, "Failed to bind to media container service");                            params.serviceError();                            return;                        } else {                            // Once we bind to the service, the first                            // pending request will be processed.                            mPendingInstalls.add(idx, params);                        }                    } else {                        mPendingInstalls.add(idx, params);                        // Already bound to the service. Just make                        // sure we trigger off processing the first request.                        if (idx == 0) {                            mHandler.sendEmptyMessage(MCS_BOUND);                        }                    }                    break;                }                case MCS_BOUND: {                    if (DEBUG_INSTALL) Slog.i(TAG, "mcs_bound");                    if (msg.obj != null) {                        mContainerService = (IMediaContainerService) msg.obj;                    }                    if (mContainerService == null) {                        // Something seriously wrong. Bail out                        Slog.e(TAG, "Cannot bind to media container service");                        for (HandlerParams params : mPendingInstalls) {                            // Indicate service bind error                            params.serviceError();                        }                        mPendingInstalls.clear();                    } else if (mPendingInstalls.size() > 0) {                        HandlerParams params = mPendingInstalls.get(0);//獲取第一個任務                        if (params != null) {                            if (params.startCopy()) {//開始安裝                                // We are done...  look for more work or to                                // go idle.                                if (DEBUG_SD_INSTALL) Log.i(TAG,                                        "Checking for more work or unbind...");                                // Delete pending install                                if (mPendingInstalls.size() > 0) {                                    mPendingInstalls.remove(0);                                }                                if (mPendingInstalls.size() == 0) {                                    if (mBound) {                                        if (DEBUG_SD_INSTALL) Log.i(TAG,                                                "Posting delayed MCS_UNBIND");                                        removeMessages(MCS_UNBIND);                                        Message ubmsg = obtainMessage(MCS_UNBIND);                                        // Unbind after a little delay, to avoid                                        // continual thrashing.                                        sendMessageDelayed(ubmsg, 10000);                                    }                                } else {                                    // There are more pending requests in queue.                                    // Just post MCS_BOUND message to trigger processing                                    // of next pending install.                                    if (DEBUG_SD_INSTALL) Log.i(TAG,                                            "Posting MCS_BOUND for next work");                                    mHandler.sendEmptyMessage(MCS_BOUND);                                }                            }                        }                    } else {                        // Should never happen ideally.                        Slog.w(TAG, "Empty queue");                    }                    break;                }                ...         }     }...} 首先,sendMessage發出的消息由PackageHandler.handleMessage()捕獲,而具體的執行由PackageHandler.doHandleMessage()完成,接著,調用抽象類HandlerParams中的一個startCopy()方法,進而調用scanPackageLI()。


上一篇:Unity問題集合

下一篇:修改Linux IO調度器

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
97av在线影院| 91高潮在线观看| 国产精品久久久久7777婷婷| 久久影院资源站| 国产视频久久网| 日韩影视在线观看| 色偷偷av一区二区三区| 欧美成人免费全部| 97视频在线观看亚洲| 日韩在线观看免费网站| 日韩精品免费看| 日韩av三级在线观看| 国产精品96久久久久久| 日韩av男人的天堂| 亚洲国产日韩欧美综合久久| 亚洲精品影视在线观看| 国产精品久久久久久久午夜| 热re99久久精品国产66热| 91美女福利视频高清| 国产国语刺激对白av不卡| 日韩美女视频中文字幕| 欧美在线影院在线视频| 国产综合视频在线观看| 日韩av在线免费观看一区| 欧洲日韩成人av| 中文综合在线观看| 疯狂做受xxxx高潮欧美日本| 国产精品久久久久7777婷婷| 在线精品国产成人综合| 国产精品对白刺激| 国产网站欧美日韩免费精品在线观看| 国产精品福利在线观看网址| 91人成网站www| 精品二区三区线观看| 亚洲老板91色精品久久| 午夜精品视频网站| 欧美日韩国产二区| 国产精品免费久久久| 国产欧美精品日韩精品| 欧美一级片在线播放| 18一19gay欧美视频网站| 日本国产高清不卡| 国产在线视频不卡| 国产精品久久不能| 青青草国产精品一区二区| 亚洲天堂男人天堂女人天堂| 国产精品丝袜久久久久久高清| 日韩成人在线网站| 久久久91精品国产| xvideos国产精品| 精品一区二区三区四区在线| 日韩av免费观影| 日韩欧美亚洲一二三区| 欧美日韩在线另类| 久久久国产精彩视频美女艺术照福利| 国产日韩在线播放| 欧美日韩国产在线| 国产精品男人的天堂| 91精品啪aⅴ在线观看国产| 亚洲美女av黄| 成人两性免费视频| 日韩成人久久久| 国产一区二区激情| 亚洲乱码一区av黑人高潮| 亚洲男人的天堂网站| 68精品国产免费久久久久久婷婷| 国产精品va在线| 国产精品wwww| 欧美日韩aaaa| 一区国产精品视频| 97在线精品视频| 欧美精品成人在线| 91色在线视频| 亚洲欧洲成视频免费观看| 午夜精品久久久久久99热软件| 日韩av手机在线观看| 亚洲高清一区二| 亚洲电影免费在线观看| 成人午夜激情免费视频| 欧美在线视频观看| 亚洲一区二区三区乱码aⅴ蜜桃女| 成人在线激情视频| 亚洲一区二区福利| 亚洲第一精品夜夜躁人人爽| 亚洲成年人影院在线| 97久久超碰福利国产精品…| 欧美一区二三区| 秋霞成人午夜鲁丝一区二区三区| 黄色精品在线看| 91精品国产沙发| 丝袜美腿精品国产二区| 国产精品女主播视频| 欧美日韩精品国产| 精品亚洲一区二区| 亚洲国产精品va在线看黑人动漫| 91日韩在线视频| 久久69精品久久久久久久电影好| 亚洲第一国产精品| 国产精品成人在线| 日韩专区在线观看| 久久久久久久久久久免费精品| 日韩欧美亚洲一二三区| 欧美日韩在线看| 97精品视频在线观看| 国产精品自产拍在线观看| www欧美日韩| 精品国产欧美成人夜夜嗨| 国产中文字幕日韩| 久久成人精品视频| 91精品国产综合久久香蕉922| 日韩欧美在线视频免费观看| 伊人男人综合视频网| 国产精品自产拍在线观| 这里精品视频免费| 亚洲欧美色图片| 成人福利网站在线观看| 久久亚洲精品国产亚洲老地址| 57pao成人永久免费视频| 久久亚洲欧美日韩精品专区| 亚洲国产精品久久久久| 日韩女优人人人人射在线视频| 国产在线视频欧美| 成人在线视频网站| 欧美日韩成人在线视频| 日韩中文字幕网| 亚洲精品午夜精品| 日韩av影视在线| 欧美在线精品免播放器视频| 国产日本欧美在线观看| 久久久精品2019中文字幕神马| 欧美激情精品久久久久久大尺度| 91精品视频在线看| 成人久久18免费网站图片| 久久亚洲精品毛片| 久久在线免费观看视频| 久久频这里精品99香蕉| 日韩欧美在线网址| 国产精品白嫩初高中害羞小美女| 久久777国产线看观看精品| 色综合色综合网色综合| 欧美成人精品一区二区| 欧美黄色免费网站| 91精品综合视频| 日韩中文字幕在线免费观看| 国产精品久久久久影院日本| 国产欧美中文字幕| 国产精品男人的天堂| 国产97人人超碰caoprom| 在线视频中文亚洲| 91精品国产91久久久久久不卡| 国产一区二区三区视频在线观看| 成人h片在线播放免费网站| 欧美尺度大的性做爰视频| 亚洲精品国产成人| 欧美与黑人午夜性猛交久久久| 在线精品播放av| 亚洲精品按摩视频| 精品久久久久久国产91| 亚洲成人网在线| 亚洲国产另类 国产精品国产免费| 久久久久久国产精品美女| 久久国内精品一国内精品| 日韩国产欧美精品一区二区三区| 国产日本欧美一区二区三区在线|