日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

詳解Android中的ActivityThread和APP啟動過程

瀏覽:171日期:2022-09-18 10:16:01
目錄ActiviryThreadActivityThread的初始化主線程Looper的初始化主線程Handler的初始化ApplicationThread及Activity的創建和啟動APP的啟動系統的啟動過程APP的啟動過程APP啟動過程的部分代碼思考總結ActiviryThreadActivityThread的初始化

ActivityThread即Android的主線程,也就是UI線程,ActivityThread的main方法是一個APP的真正入口,MainLooper在它的main方法中被創建。

//ActivityThread的main方法public static void main(String[] args) { ... Looper.prepareMainLooper(); ActivityThread thread = new ActivityThread(); //在attach方法中會完成Application對象的初始化,然后調用Application的onCreate()方法 thread.attach(false); if (sMainThreadHandler == null) {sMainThreadHandler = thread.getHandler(); } ... Looper.loop(); throw new RuntimeException('Main thread loop unexpectedly exited');}

接下來從主線程Looper的初始化和ApplicationThread及Activity的創建啟動兩方面,通過源碼了解學習下大致的流程。

主線程Looper的初始化

Looper.prepareMainLooper();相關的代碼如下

//主線程Looper的初始化public static void prepareMainLooper() { prepare(false); synchronized (Looper.class) {if (sMainLooper != null) { throw new IllegalStateException('The main Looper has already been prepared.');}sMainLooper = myLooper(); }}//普通線程Looper的初始化public static void prepare() { prepare(true);}private static void prepare(boolean quitAllowed) { if (sThreadLocal.get() != null) {throw new RuntimeException('Only one Looper may be created per thread'); } sThreadLocal.set(new Looper(quitAllowed));}

看過Handler源碼就知道,主線程Looper的初始化和普通線程Looper的初始化很相似,但還是有以下幾個區別

1.普通線程的Prepare()默認quitAllowed參數為true,表示允許退出,而主線程也就是ActivityThread的Looper參數為false,不允許退出。這里的quitAllowed參數,最終會傳遞給MessageQueue,當調用MessageQueue的quit方法時,會判斷這個參數,如果是主線程,也就是quitAllowed參數為false時,會拋出異常。

//Looper的退時會判斷quitAllowedvoid quit(boolean safe) { if (!mQuitAllowed) {throw new IllegalStateException('Main thread not allowed to quit.'); } synchronized (this) {... }}

2.我們注意到主線程Looper初始化之后,賦值給了成員變量sMainLooper,這個成員的作用就是向其他線程提供主線程的Looper對象。這下我們就應該知道為什么Looper.getMainLooper()方法能獲取主線程的Looper對象了

public static Looper getMainLooper() { synchronized (Looper.class) {return sMainLooper; }}主線程Handler的初始化

在ActivityThread的main方法中我們注意到一行代碼:

ActivityThread thread = new ActivityThread();thread.attach(false);if (sMainThreadHandler == null) { sMainThreadHandler = thread.getHandler();}

見名知意,這是獲取主線程的Handler,那么主線程的Handler是在什么時候初始化的呢?

//與之相關的代碼如下://ActivityThread的成員變量final H mH = new H();final Handler getHandler() { return mH;}

從以上代碼中可以看到,主線程的Handler作為ActivityThread的成員變量,是在ActivityThread的main方法被執行,ActivityThread被創建時而初始化,而接下來要說的ApplicationThread中的方法執行以及Activity的創建都依賴于主線程Handler。至此我們也就明白了,主線程(ActivityThread)的初始化是在它的main方法中,主線程的Handler以及MainLooper的初始化時機都是在ActivityThread創建的時候。

ApplicationThread及Activity的創建和啟動

以上的代碼和流程,就是對 MainLooper 和 ActivityThread 的初始化,我們接下來看一下 ActivityThread 的初始化及其對應的 attach 方法,在thread.attach方法中,ActivityManagerService通過attachApplication方法,將ApplicationThread對象綁定到ActivityManagerService,ApplicationThread是ActivityThread的私有內部類,實現了IBinder接口,用于ActivityThread和ActivityManagerService的所在進程間通信。

//ActivityThread的attach方法:private void attach(boolean system) { ... if (!system) {final IActivityManager mgr = ActivityManager.getService();try { mgr.attachApplication(mAppThread);} catch (RemoteException ex) { throw ex.rethrowFromSystemServer();}else{ ...} }}//ActivityManagerService中的方法:public final void attachApplication(IApplicationThread thread) { synchronized (this) {int callingPid = Binder.getCallingPid();final long origId = Binder.clearCallingIdentity();attachApplicationLocked(thread, callingPid);Binder.restoreCallingIdentity(origId); }}

這里的個人理解是:在每個ActivityThread(APP)被創建的時候,都需要向ActivityManagerService綁定(或者說是向遠程服務AMS注冊自己),用于AMS管理ActivityThread中的所有四大組件的生命周期。

上述AMS的代碼中attachApplicationLocked方法比較復雜,主要功能有兩個,詳見注釋,這里忽略了很多代碼細節,具體的流程可以看源碼

//AMS中的方法,主要功能有以下兩步private final boolean attachApplicationLocked(IApplicationThread thread, int pid) { ... //主要用于創建Application,用調用onCreate方法 thread.bindApplication(...); ... //主要用于創建Activity if (mStackSupervisor.attachApplicationLocked(app)) {... }}

1.thread.bindApplication:主要用于創建Application,這里的thread對象是ApplicationThread在AMS中的代理對象,所以這里的bindApplication方法最終會調用ApplicationThread.bindApplication()方法,該方法會向ActivityThread的消息對應發送BIND_APPLICATION的消息,消息的處理最終會調用Application.onCreate()方法,這也說明Application.onCreate()方法的執行時機比任何Activity.onCreate()方法都早。

//ActivityThread中的bindApplication方法public final void bindApplication(...) { ... // 該消息的處理,會調用handleBindApplication方法 sendMessage(H.BIND_APPLICATION, data);}//ActivityThread中的handleBindApplication方法private void handleBindApplication(AppBindData data) { ... try {Application app = data.info.makeApplication(data.restrictedBackupMode, null);mInitialApplication = app;...try { mInstrumentation.callApplicationOnCreate(app);} catch (Exception e) {} } finally { }} //LoadedApk中的方法,用于創建Applicationpublic Application makeApplication(boolean forceDefaultAppClass, Instrumentation instrumentation) { //如果存在Application的實例,則直接返回,這也說明Application是個單例 if (mApplication != null) {return mApplication; } Application app = null; //...這里通過反射初始化Application if (instrumentation != null) {try { //調用Application的onCreate方法 instrumentation.callApplicationOnCreate(app);} catch (Exception e) {} } return app;}

2.mStackSupervisor.attachApplicationLocked(app):用于創建Activity,mStackSupervisor是AMS的成員變量,為Activity堆棧管理輔助類實例,該方法最終會調用ApplicationThread類的scheduleLaunchActivity方法,該方法也是類似于第一步,向ActivityThread的消息隊列發送創建Activity的消息,最終在ActivityThread中完成創建Activity的操作。

boolean attachApplicationLocked(ProcessRecord app) throws RemoteException { ... if (realStartActivityLocked(hr, app, true, true)) {... } ...}final boolean realStartActivityLocked(ActivityRecord r, ProcessRecord app, boolean andResume, boolean checkConfig) throws RemoteException { ... try {//調用ApplicationThread的scheduleLaunchActivity用于啟動一個Activityapp.thread.scheduleLaunchActivity(...); } catch (RemoteException e) { }}

ApplicationThread的scheduleLaunchActivity方法會向ActivityThread發送LAUNCH_ACTIVITY信息,用于啟動一個Activity,該消息的處理會調用ActivityThread的handleLaunchActivity方法,最終啟動一個Activity

以上就是從ActivityThread的main方法執行到Activity的創建之間的流程,至于ActivityThread的main方法執行時機,以及執行前的流程和Activity的具體創建過程,可以接著看APP的啟動過程

APP的啟動系統的啟動過程

在學習APP的啟動之前先簡單了解下系統的啟動,有助于我們更好的學習APP的啟動。系統的啟動過程很復雜,這里簡單化,只關心大致流程和涉及到的一些名詞以及相關類的作用

APP的啟動可以簡單總結為一下幾個流程:

加載BootLoader --> 初始化內核 --> 啟動init進程 --> init進程fork出Zygote進程 --> Zygote進程fork出SystemServer進程

詳解Android中的ActivityThread和APP啟動過程

系統中的所有經常進程都是由Zygote進程fork出來的 SystemServer進程是系統進程,很多系統服務,例如ActivityManagerService、PackageManagerService、WindowManagerService…都是存在該進程被創建后啟動 ActivityManagerServices(AMS):是一個服務端對象,負責所有的Activity的生命周期,AMS通過Binder與Activity通信,而AMS與Zygote之間是通過Socket通信 ActivityThread:本篇的主角,UI線程/主線程,它的main()方法是APP的真正入口 ApplicationThread:一個實現了IBinder接口的ActivityThread內部類,用于ActivityThread和AMS的所在進程間通信 Instrumentation:可以理解為ActivityThread的一個工具類,在ActivityThread中初始化,一個進程只存在一個Instrumentation對象,在每個Activity初始化時,會通過Activity的Attach方法,將該引用傳遞給Activity。Activity所有生命周期的方法都有該類來執行APP的啟動過程

APP的啟動,我們使用一張圖來說明這個啟動過程,順便也總結下上面所說的ActivityThread的main方法執行到Activity的創建之間的流程。

詳解Android中的ActivityThread和APP啟動過程

1.點擊桌面APP圖標時,Launcher的startActivity()方法,通過Binder通信,調用system_server進程中AMS服務的startActivity方法,發起啟動請求

2.system_server進程接收到請求后,向Zygote進程發送創建進程的請求

3.Zygote進程fork出App進程,并執行ActivityThread的main方法,創建ActivityThread線程,初始化MainLooper,主線程Handler,同時初始化ApplicationThread用于和AMS通信交互

4.App進程,通過Binder向sytem_server進程發起attachApplication請求,這里實際上就是APP進程通過Binder調用sytem_server進程中AMS的attachApplication方法,上面我們已經分析過,AMS的attachApplication方法的作用是將ApplicationThread對象與AMS綁定

5.system_server進程在收到attachApplication的請求,進行一些準備工作后,再通過binder IPC向App進程發送handleBindApplication請求(初始化Application并調用onCreate方法)和scheduleLaunchActivity請求(創建啟動Activity)

6.App進程的binder線程(ApplicationThread)在收到請求后,通過handler向主線程發送BIND_APPLICATION和LAUNCH_ACTIVITY消息,這里注意的是AMS和主線程并不直接通信,而是AMS和主線程的內部類ApplicationThread通過Binder通信,ApplicationThread再和主線程通過Handler消息交互。 ( 這里猜測這樣的設計意圖可能是為了統一管理主線程與AMS的通信,并且不向AMS暴露主線程中的其他公開方法)

7.主線程在收到Message后,創建Application并調用onCreate方法,再通過反射機制創建目標Activity,并回調Activity.onCreate()等方法

8.到此,App便正式啟動,開始進入Activity生命周期,執行完onCreate/onStart/onResume方法,UI渲染后顯示APP主界面

APP啟動過程的部分代碼思考

在上面學習APP的啟動過程中,看源碼的同時注意到一個代碼,就是主線程Handler在接收到LAUNCH_ACTIVITY創建Activity的消息后,創建Activity的部分代碼如下:

//主線程Handler接收到創建Activity的消息LAUNCH_ACTIVITY后,最終會調用performLaunchActivity方法//performLaunchActivity方法會通過反射去創建一個Activity,然后會調用Activity的各個生命周期方法private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) { ... ContextImpl appContext = createBaseContextForActivity(r); Activity activity = null; try {//這里是反射創建Activityjava.lang.ClassLoader cl = appContext.getClassLoader();activity = mInstrumentation.newActivity(cl, component.getClassName(), r.intent);StrictMode.incrementExpectedActivityCount(activity.getClass());r.intent.setExtrasClassLoader(cl);r.intent.prepareToEnterProcess();if (r.state != null) { r.state.setClassLoader(cl);} } try {//這里注意,又調用了一次Application的創建方法,但是前面分析過,Application是個單例,所以這里的實際上是獲取Application實例,但是這里為什么會再次調用創建Application的方法呢?Application app = r.packageInfo.makeApplication(false, mInstrumentation);... } ... return activity;}

在上面的代碼中,簡單注釋了一下在Activity的創建方法中,會再次調用Application的創建方法(第一次調用是在接收到BIND_APPLICATION消息的時候),個人覺得這里再次調用Application的創建方法,除了獲取已經存在的Application實例這種情況,另外一種情況還有可能是要創建的這個Activity屬于另外一個進程,當去啟動這個新進程中的Activity時,會先去創建新進程和Application實例,因為我們知道一個常識:

1.APP中有幾個進程,Application會被創建幾次

2.新進程中所有變量和單例會失效,因為新進程有一塊新的內存區域

那么這兩點的關系就是,因為新進程中Application實例會為空,所以會再次去創建Application實例,這也就是第一點中我們所說的常識:APP中有幾個進程,Application會被創建幾次

//創建Application的方法public Application makeApplication(boolean forceDefaultAppClass, Instrumentation instrumentation) { //如果存在Application的實例,則直接返回,這也說明Application是個單例 if (mApplication != null) {return mApplication; } Application app = null; //...創建Application return app;}

那么依次類推,Service作為四大組件之一,類似于Activity的創建和啟動,創建Service的方法中會不會也調用了創建Application的方法(makeApplication方法),答案是肯定的!和Activity的創建類似,當我們調用startService的時候,也是通過Binder向AMS發送創建Service的請求,AMS準備后再向APP進程發送scheduleCreateService的請求,然后主線程handle收到CREATE_SERVICE的消息,調用handleCreateService創建Service的方法。在創建Service的方法handleCreateService中也調用了創建Application的方法,具體代碼看源碼吧。所以我們也徹底明白了為什么APP中有幾個進程,Application會被創建幾次,以及Application為什么是個單例。

總結

APP的啟動過程很復雜,代碼錯綜交橫,這里分析了大概流程,學習這部分的過程中還是有很多收獲,例如知道了AMS與主線程的關系,主線程main方法中就是APP的入口,Binder通信機制和handler消息機制在這個過程中的重要作用,Application的創建時機以及Application為什么是單例,為什么有幾個進程就創建幾個Application…等等 。

以上就是詳解Android中的ActivityThread和APP啟動過程的詳細內容,更多關于Android ActivityThread APP啟動過程的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
精品一区二区三区中文字幕在线| 中文视频一区| 亚洲丝袜啪啪| 日本中文字幕不卡| 五月天激情综合网| 欧美在线影院| 日韩精品导航| 精品国产中文字幕第一页| 日产精品一区二区| 999国产精品| 免费欧美一区| 丝袜av一区| 国产精品字幕| 亚洲激情国产| 奇米狠狠一区二区三区| 日韩高清在线不卡| 欧美激情视频一区二区三区免费| 国产精品久久久久毛片大屁完整版| 日本不卡一二三区黄网| 亚洲一区久久| 国产日韩一区二区三免费高清| 国产精品精品| 石原莉奈在线亚洲二区| 日韩va亚洲va欧美va久久| 日本视频中文字幕一区二区三区| 亚洲精品麻豆| 日韩精品第一区| 91高清一区| 国产探花一区在线观看| 尤物网精品视频| 国产aⅴ精品一区二区三区久久 | 精品欧美日韩精品| 国产精品伦理久久久久久| 国产精品高颜值在线观看| 日韩在线一二三区| 91精品国产91久久久久久黑人| 中文无码久久精品| 电影天堂国产精品| 久久激情综合网| 亚洲欧美视频| 国产一区观看| 国产精品伦一区二区| 蜜桃视频欧美| 91亚洲国产| 国产精品一在线观看| 中文字幕乱码亚洲无线精品一区| 日韩欧美在线中字| 久久精品97| 精品一区欧美| 久久婷婷激情| 国产一区二区三区久久久久久久久| 中文字幕免费精品| 蜜臀91精品一区二区三区| 成人国产精品| 久久久久久久欧美精品| 不卡专区在线| 国产高潮在线| 另类综合日韩欧美亚洲| 国产美女久久| 久久国产免费看| 国产精品22p| 国产精品超碰| 国产精品v亚洲精品v日韩精品| 日本aⅴ亚洲精品中文乱码| 亚洲色图网站| 日韩一区二区三免费高清在线观看 | 天堂精品久久久久| 91国内精品| 美女国产精品久久久| 成人午夜亚洲| 久久精品国产大片免费观看| 色婷婷久久久| 丝袜美腿亚洲色图| 日本 国产 欧美色综合| 蜜臀a∨国产成人精品| 久久久久国产精品一区二区| 制服诱惑一区二区| 日韩国产欧美一区二区三区| 国产精久久久| 亚洲特色特黄| 国产午夜久久av| 92国产精品| 日韩av午夜在线观看| 成人一区而且| 免费视频最近日韩| 粉嫩av一区二区三区四区五区 | 国产极品嫩模在线观看91精品| 福利一区和二区| 尤物网精品视频| 国产中文字幕一区二区三区| 国产亚洲综合精品| 成人日韩av| 蜜臀久久99精品久久久久宅男| 国产精品尤物| 99在线精品免费视频九九视| 亚洲青青久久| 狠狠久久伊人中文字幕| 亚洲一区免费| 色在线视频观看| 欧美日韩va| 久久xxxx| 国产一区二区视频在线看| 蜜桃视频在线观看一区| 久久电影tv| 久久尤物视频| 日韩在线成人| 免费在线观看日韩欧美| 日韩精品一区二区三区免费观影 | 麻豆成人91精品二区三区| 色婷婷久久久| 国产精品毛片久久| 国产精品va| 欧美日韩精品一区二区三区视频| 99香蕉国产精品偷在线观看 | 亚洲国产一区二区三区在线播放| 国产精品一区高清| 日韩精品一页| 美女日韩在线中文字幕| 桃色一区二区| 亚洲va中文在线播放免费| 久久中文字幕一区二区三区| 欧美国产亚洲精品| 久久精品97| 精品视频一二| 丰满少妇一区| 三级在线看中文字幕完整版| 激情国产在线| 激情偷拍久久| av中文资源在线资源免费观看| 国内精品伊人| 日韩av二区| av在线最新| 999久久久91| 久久午夜影视| 日本在线成人| 久久国产欧美日韩精品| 青青在线精品| 麻豆精品新av中文字幕| 精品福利久久久| 国产成人久久精品麻豆二区 | 免费观看久久久4p| 日本aⅴ亚洲精品中文乱码 | 激情久久中文字幕| 国产伊人精品| 中文不卡在线| 老司机免费视频一区二区三区| 你懂的国产精品永久在线| 精品不卡一区| 亚洲精品97| 在线亚洲国产精品网站| 亚洲日本久久| 免费在线观看一区二区三区| 视频一区在线视频| 久久国内精品| 久久一区二区三区电影| 日韩中文字幕一区二区三区| 青草国产精品久久久久久| 国产在线一区不卡| 夜久久久久久| 欧美1区2区3| 免播放器亚洲| 色婷婷亚洲mv天堂mv在影片| 国模大尺度视频一区二区| 久久国产直播| 精品欧美久久| 国产精品v一区二区三区| 久久男女视频| 亚洲一区二区三区久久久| 欧美一区二区三区久久| 国产成人精选| 国产欧美日韩免费观看| 久久国产电影| 久久亚洲黄色| 日韩高清二区| 激情欧美丁香| 伊人网在线播放| 国产精品伦一区二区| 亚洲视频二区| 首页欧美精品中文字幕| 一区二区三区四区日本视频| 日韩精品第一| 一区二区精彩视频| 在线一区视频观看| 国产精品久久久一区二区| 日韩精品一区二区三区中文| 欧美午夜不卡影院在线观看完整版免费| 国产精品大片| 国产精品15p| 你懂的国产精品永久在线| 久久精品av麻豆的观看方式| 蜜桃传媒麻豆第一区在线观看 | 蜜臀国产一区二区三区在线播放| 欧美理论视频| 久久一级电影| 欧美日韩国产综合网| 亚洲涩涩在线| 日韩免费看片| 久久久久国产一区二区| 久久91导航| 99国产精品私拍|