APP啟動流程(一)Launcher到ActivityTaskManagerService

前置知識

  • 當(dāng)Android系統(tǒng)啟動時,首先會創(chuàng)建Zygote進程,Zygote進程啟動后會fork若干其他的進程摹蘑,例如SystemServer進程,Launcher進程轧飞。

  • 創(chuàng)建SystemServer進程后會創(chuàng)建SystemServiceManager衅鹿,啟動很多系統(tǒng)服務(wù),例ActivityTaskManagerService踪少,ActivityManagerService恢着,PackageManagerService等源哩。

  • 其中Launcher桌面實際上就是一個Activity,當(dāng)我們點擊桌面icon,也是通過startActivity來啟動App的磅崭。

在桌面點擊一個APP圖標(biāo)啟動APP的過程中譬涡,涉及到了跨進程通信合呐,APP進程創(chuàng)建板乙,Application和Activity的創(chuàng)建和啟動等內(nèi)容。

本篇文章主要分析點擊桌面App圖標(biāo)到ActivityTaskManagerService的過程具篇。

涉及到的幾個類

  • packages/apps/Launcher3/src/com/android/launcher3/Launcher.java
  • frameworks/base/core/java/android/app/Instrumentation.java
  • frameworks/base/services/core/java/com/android/server/wm/ActivityTaskManagerService.java

具體過程:

1.找到Launcher桌面的APP圖標(biāo)入口位置

  1. Launcher是一個Activity纬霞,加載的布局為R.layout.launcher
  2. launcher.xml中的layout="@layout/all_apps"引入了LauncherAllAppsContainerView,其中layout="@layout/all_apps_rv_layout"引入了AllAppsRecyclerView
  3. 在AllAppsContainerView中為其設(shè)置的adapter是AllAppsGridAdapter驱显,點擊事件由launcher.getItemOnClickListener()獲取诗芜,通過ItemClickHandler.INSTANCE單例方法獲取點擊事件
  4. 最終調(diào)用ItemClickHandler的onClick->startAppShortcutOrInfoActivity->launcher.startActivitySafely

packages/apps/Launcher3/src/com/android/launcher3/touch/ItemClickHandler.java

private static void onClick(View v, String sourceContainer) {
    if (v.getWindowToken() == null) return;
    Launcher launcher = Launcher.getLauncher(v.getContext());
    if (!launcher.getWorkspace().isFinishedSwitchingState()) return;
    Object tag = v.getTag();
    if (tag instanceof WorkspaceItemInfo) {
        onClickAppShortcut(v, (WorkspaceItemInfo) tag, launcher, sourceContainer);
    } else if (tag instanceof FolderInfo) {
        if (v instanceof FolderIcon) {
            onClickFolderIcon(v);
        }
    } else if (tag instanceof AppInfo) {
        //點擊App圖標(biāo)
        startAppShortcutOrInfoActivity(v, (AppInfo) tag, launcher,
                sourceContainer == null ? CONTAINER_ALL_APPS: sourceContainer);
    } else if (tag instanceof LauncherAppWidgetInfo) {
        if (v instanceof PendingAppWidgetHostView) {
            onClickPendingWidget((PendingAppWidgetHostView) v, launcher);
        }
    }
}

private static void startAppShortcutOrInfoActivity(View v, ItemInfo item, Launcher launcher,
            @Nullable String sourceContainer) {
    Intent intent;
    if (item instanceof PromiseAppInfo) {
        PromiseAppInfo promiseAppInfo = (PromiseAppInfo) item;
        intent = promiseAppInfo.getMarketIntent(launcher);
    } else {
        intent = item.getIntent();
    }
    if (intent == null) {
        throw new IllegalArgumentException("Input must have a valid intent");
    }
    if (item instanceof WorkspaceItemInfo) {
        WorkspaceItemInfo si = (WorkspaceItemInfo) item;
        if (si.hasStatusFlag(WorkspaceItemInfo.FLAG_SUPPORTS_WEB_UI)
                && Intent.ACTION_VIEW.equals(intent.getAction())) {
            intent = new Intent(intent);
            intent.setPackage(null);
        }
    }
    if (v != null && launcher.getAppTransitionManager().supportsAdaptiveIconAnimation()) {
        FloatingIconView.fetchIcon(launcher, v, item, true /* isOpening */);
    }
    //調(diào)用
    launcher.startActivitySafely(v, intent, item, sourceContainer);
}

2.startActivitySafely->startActivityForResult

packages/apps/Launcher3/src/com/android/launcher3/Launcher.java

public boolean startActivitySafely(View v, Intent intent, @Nullable ItemInfo item, @Nullable String sourceContainer) {
    //...
    //添加FLAG_ACTIVITY_NEW_TASK
    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    try {
        boolean isShortcut = (item instanceof WorkspaceItemInfo)
                && (item.itemType == Favorites.ITEM_TYPE_SHORTCUT
                || item.itemType == Favorites.ITEM_TYPE_DEEP_SHORTCUT)
                && !((WorkspaceItemInfo) item).isPromise();
        if (isShortcut) {
            //點擊Shortcut啟動activity
            // Shortcuts need some special checks due to legacy reasons.
            startShortcutIntentSafely(intent, optsBundle, item, sourceContainer);
        } else if (user == null || user.equals(Process.myUserHandle())) {
            //點擊app圖標(biāo)啟動app
            // Could be launching some bookkeeping activity
            startActivity(intent, optsBundle);
            AppLaunchTracker.INSTANCE.get(this).onStartApp(intent.getComponent(),
                    Process.myUserHandle(), sourceContainer);
        } else {
            getSystemService(LauncherApps.class).startMainActivity(
                    intent.getComponent(), user, intent.getSourceBounds(), optsBundle);
            AppLaunchTracker.INSTANCE.get(this).onStartApp(intent.getComponent(), user,
                    sourceContainer);
        }
        return true;
    } catch (NullPointerException|ActivityNotFoundException|SecurityException e) {
    ...
    }
    return false;
}


public void startActivity(Intent intent, @Nullable Bundle options) {
    //本質(zhì)上還是通過startActivityForResult啟動activity
    if (options != null) {
        startActivityForResult(intent, -1, options);
    } else {
        startActivityForResult(intent, -1);
    }
}
    
public void startActivityForResult(@RequiresPermission Intent intent, int requestCode, @Nullable Bundle options) {
    //此時需要啟動的Activity的parent還是null
    if (mParent == null) {
        options = transferSpringboardActivityOptions(options);
        //Instrumentation執(zhí)行了execStartActivity方法
        //mMainThread:Launcher所在的進程
        //mMainThread.getApplicationThread():獲取mMainThread.getApplicationThread(),ApplicationThread 繼承自 Stub埃疫,也就是 Binder伏恐,用于進程間通信
        Instrumentation.ActivityResult ar =
            mInstrumentation.execStartActivity(
                this, mMainThread.getApplicationThread(), mToken, this,
                intent, requestCode, options);
        if (ar != null) {
            mMainThread.sendActivityResult(
                mToken, mEmbeddedID, requestCode, ar.getResultCode(),
                ar.getResultData());
        }
        if (requestCode >= 0) {
            mStartedActivity = true;
        }

        cancelInputsAndStartExitTransition(options);
    } else {
        if (options != null) {
            mParent.startActivityFromChild(this, intent, requestCode, options);
        } else {
            mParent.startActivityFromChild(this, intent, requestCode);
        }
    }
}

3.Instrumentation.execStartActivity->ActivityTaskManagerService.startActivity

frameworks/base/core/java/android/app/Instrumentation.java

Instrumentation是系統(tǒng)進程和App進程交互的監(jiān)測類,執(zhí)行execStartActivity

public ActivityResult execStartActivity(
        Context who, IBinder contextThread, IBinder token, Activity target,
        Intent intent, int requestCode, Bundle options) {
    IApplicationThread whoThread = (IApplicationThread) contextThread;
    Uri referrer = target != null ? target.onProvideReferrer() : null;
    try {
        intent.migrateExtraStreamToClipData(who);
        intent.prepareToLeaveProcess(who);
        //通過binder機制栓霜,ActivityTaskManager.getService()獲取到了ActivityTaskManagerservice
        int result = ActivityTaskManager.getService().startActivity(whoThread,
                who.getBasePackageName(), who.getAttributionTag(), intent,
                intent.resolveTypeIfNeeded(who.getContentResolver()), token,
                target != null ? target.mEmbeddedID : null, requestCode, 0, null, options);
        checkStartActivityResult(result, intent);
    } catch (RemoteException e) {
        throw new RuntimeException("Failure from system", e);
    }
    return null;
}

frameworks/base/services/core/java/com/android/server/wm/ActivityTaskManagerService.java

ActivityTaskManagerService繼承了IActivityTaskManager.Stub翠桦,這里采用了AIDL的形式獲取到了ActivityTaskManagerService的代理類
,然后跨進程通信ActivityTaskManagerService調(diào)用startActivity方法啟動activity

public static IActivityTaskManager getService() {
    return IActivityTaskManagerSingleton.get();
}

@UnsupportedAppUsage(trackingBug = 129726065)
private static final Singleton<IActivityTaskManager> IActivityTaskManagerSingleton =
        new Singleton<IActivityTaskManager>() {
            @Override
            protected IActivityTaskManager create() {
                //通過 ServiceManager 獲取對應(yīng)的系統(tǒng)服務(wù)胳蛮,也就是 IBinder 類型的 ATMS 引用销凑。
                final IBinder b = ServiceManager.getService(Context.ACTIVITY_TASK_SERVICE);
                return IActivityTaskManager.Stub.asInterface(b);
            }
        };

ActivityTaskManagerService extends IActivityTaskManager.Stub

總結(jié):startActivitySafely(Launcher)->startActivityForResult(Activity)->execStartActivity(Instrumentation)->startActivity(ATMS)

參考:Android Code Search

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末丛晌,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子斗幼,更是在濱河造成了極大的恐慌澎蛛,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,451評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件孟岛,死亡現(xiàn)場離奇詭異瓶竭,居然都是意外死亡督勺,警方通過查閱死者的電腦和手機渠羞,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評論 3 394
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來智哀,“玉大人次询,你說我怎么就攤上這事〈山校” “怎么了屯吊?”我有些...
    開封第一講書人閱讀 164,782評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長摹菠。 經(jīng)常有香客問我盒卸,道長,這世上最難降的妖魔是什么次氨? 我笑而不...
    開封第一講書人閱讀 58,709評論 1 294
  • 正文 為了忘掉前任蔽介,我火速辦了婚禮,結(jié)果婚禮上煮寡,老公的妹妹穿的比我還像新娘虹蓄。我一直安慰自己,他們只是感情好幸撕,可當(dāng)我...
    茶點故事閱讀 67,733評論 6 392
  • 文/花漫 我一把揭開白布薇组。 她就那樣靜靜地躺著,像睡著了一般坐儿。 火紅的嫁衣襯著肌膚如雪律胀。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,578評論 1 305
  • 那天貌矿,我揣著相機與錄音炭菌,去河邊找鬼。 笑死站叼,一個胖子當(dāng)著我的面吹牛娃兽,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播尽楔,決...
    沈念sama閱讀 40,320評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼投储,長吁一口氣:“原來是場噩夢啊……” “哼第练!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起玛荞,我...
    開封第一講書人閱讀 39,241評論 0 276
  • 序言:老撾萬榮一對情侶失蹤娇掏,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后勋眯,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體婴梧,經(jīng)...
    沈念sama閱讀 45,686評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,878評論 3 336
  • 正文 我和宋清朗相戀三年客蹋,在試婚紗的時候發(fā)現(xiàn)自己被綠了塞蹭。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,992評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡讶坯,死狀恐怖番电,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情辆琅,我是刑警寧澤漱办,帶...
    沈念sama閱讀 35,715評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站婉烟,受9級特大地震影響娩井,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜似袁,卻給世界環(huán)境...
    茶點故事閱讀 41,336評論 3 330
  • 文/蒙蒙 一洞辣、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧叔营,春花似錦屋彪、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,912評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至婴谱,卻和暖如春蟹但,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背谭羔。 一陣腳步聲響...
    開封第一講書人閱讀 33,040評論 1 270
  • 我被黑心中介騙來泰國打工华糖, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人瘟裸。 一個月前我還...
    沈念sama閱讀 48,173評論 3 370
  • 正文 我出身青樓客叉,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子兼搏,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,947評論 2 355