makeApplication創(chuàng)建application中會執(zhí)行attachBaseContext(context);
創(chuàng)新互聯(lián)建站,專注為中小企業(yè)提供官網(wǎng)建設(shè)、營銷型網(wǎng)站制作、成都響應(yīng)式網(wǎng)站建設(shè)公司、展示型網(wǎng)站設(shè)計制作、成都網(wǎng)站建設(shè)等服務(wù),幫助中小企業(yè)通過網(wǎng)站體現(xiàn)價值、有效益。幫助企業(yè)快速建站、解決網(wǎng)站建設(shè)與網(wǎng)站營銷推廣問題。
installContentProviders第一個參數(shù)context,是從上面?zhèn)鬟f下來的app,也就是application。
遍歷providers列表,初始化每一個provider,都是用application的context。構(gòu)造出ContentProvider然后執(zhí)行attachInfo() 方法,attachInfo()執(zhí)行完畢會執(zhí)行onCreate()。
最后再mInstrumentation.callApplicationOnCreate(app);執(zhí)行Application的OnCreate方法。
總結(jié)
從流程上來看,符合日志打印的流程Application#attachBaseContext() → ContentProvider#attachInfo() → ContentProvider#onCreate() → Application#onCreate()
ContentProvider持有的Context也是application,具備給SDK初始化使用。
這時候第一進(jìn)程是zygote。zygote英文是受精卵的意思。android系統(tǒng)的所有進(jìn)程都是由zygote進(jìn)程fork而來。zygote最先啟動的第一個進(jìn)程是鼎鼎大名的SystemServer進(jìn)程。這個進(jìn)程包含了我們常說的三個大神級系統(tǒng)服務(wù),分別是ActivityManagerService,WindowManagerService以及PackegeManagerService。
進(jìn)程入口在ActivityThread這個類的main()方法,這個main方法類似C語言的mian方法,是一個程序入口。
這個方法會接著調(diào)用ActivityManagerNatvie(一個單例類,可以獲取ActivityManagerService的實例)的getDeafault()返回ActivityManagerService實例。
ApplicationThread是ActivityThread的內(nèi)部類,他是App和系統(tǒng)跨進(jìn)程交互的入口,它的實現(xiàn)類在客戶端進(jìn)程。
獲得了正在Binder通信的客戶端的當(dāng)前線程的id,然后和ApplicationThread對象作為參數(shù)傳入到AMS的attachApplicationLocked。
thread是ApplicationThreadProxy的對象引用,它是代理對象,先調(diào)用ApplicationThreadProxy的bindApplication方法,接著在這個方法中又調(diào)用ApplicationThreadNative的函數(shù)onTransact,然后函數(shù)onTransact中根據(jù)code找到對應(yīng)的case,最終會調(diào)用ApplicationThread的bindApplication方法。
在這里,bindApplication方法通過向ActivityThread的消息隊列發(fā)送BIND_APPLICATION消息
消息的處理調(diào)用handleBindApplication方法,handleBindApplication方法比較重要的是會調(diào)用如下方法
在執(zhí)行完bindApplication()之后進(jìn)入ActivityStackSupervisor.attachApplicationLocked(),這樣我們整個應(yīng)用進(jìn)程已經(jīng)啟動起來了。開始activity的啟動邏輯了。
這個類是一個AMS的一個棧管理類,里面存儲著ActivityStack的集合。在這個方法,會遍歷各個ActivityStack,找到前臺棧,找到里面的TopActivity。然后比較 傳進(jìn)來的ProcessRecord.processName和UID是否個和opActivity對用的ActivityRecord里面的一致。如果一致,就調(diào)用ActivityStackSupervisor.realStartAcvitiyLocked(ProcessRecord,ActivityRecord)方法。
這個方法會調(diào)用傳過來的ApplicationThread實例的ScheduelLaunchActivity(包括ActivityRecord)方法,所以真正執(zhí)行的是ActivityThread中的scheduleLaunchActivity
這個方法是跨進(jìn)程的,會把ActivityRecord同步到App進(jìn)程的ActivityRecordClient數(shù)據(jù)結(jié)構(gòu),用來后面構(gòu)造Application和Activity等。
ActivityThread接收到SystemServer進(jìn)程的消息之后會通過其內(nèi)部的Handler對象分發(fā)消息,經(jīng)過一系列的分發(fā)之后調(diào)用了ActivityThread的handleLaunchActivity方法:
接著調(diào)用PerformLaunchActivity方法和HandleLaunchActivtiy()方法。performLauncherActivity,看名字應(yīng)該就是執(zhí)行Activity的啟動操作了
1.這個方法主要是構(gòu)造Application和通過mInstrumention.newActivity()構(gòu)造Activity。
這個方法會初始化一個Window,以后詳細(xì)講,人格視圖都是附在一個window的docorView上,然后由WMS.addView顯示。
這個方法會調(diào)用Actiity的resume()方法,并且在makrVisible()里面調(diào)用WMS.addView(window),這個windows里面的docorView的contentView就是onCreate()里面setContentView(int layout)設(shè)置的contentView。
注意關(guān)于WMS.addView(window),這個系統(tǒng)服務(wù),我們下次再講,里面有一個類RootViewImpl,這個類負(fù)責(zé)管理我們contentView視圖樹的逐級繪制。
原文鏈接
你是想管理“安卓手機(jī)啟動項”?還是把自已的APk應(yīng)用程序放到“安卓啟動項”里?
如果你想管理啟動項,如禁止某一程序自啟動,ROOT后用Autostarts1 .8版程序能
設(shè)置禁用; 如果想把自已的程序設(shè)成能自啟,須要在程序里捕捉啟動消息。
翻譯自:
這篇文章講解當(dāng)用戶點擊應(yīng)用圖標(biāo)時,安卓如何啟動你的應(yīng)用。安卓系統(tǒng)做了很多幕后工作,來使得你的launch activity對用戶可見。本文通過重要階段的講解和調(diào)用序列詳細(xì)講解這一過程。
安卓應(yīng)用在這兩個方面是獨特的:
多個入口點 :Android應(yīng)用程序由不同的組件組成,它們可以調(diào)用其他應(yīng)用程序擁有的組件。這些組件大致對應(yīng)于任何應(yīng)用程序的多個入口點。因此,它們不同于具有像main()方法那樣的單個入口點的傳統(tǒng)應(yīng)用程序。
擁有自己的小世界 :每個Android應(yīng)用程序都生活在自己的世界中,它在單獨的進(jìn)程中運(yùn)行,擁有自己的Dalvik VM實例,并分配有唯一的用戶ID。
必要時會啟動Android進(jìn)程。
每當(dāng)用戶或其他系統(tǒng)組件請求執(zhí)行屬于您應(yīng)用程序的組件(可能是服務(wù),活動或意圖接收器)時,Android系統(tǒng)都會為您的應(yīng)用程序啟動一個新進(jìn)程(如果尚未運(yùn)行)。通常,進(jìn)程一直運(yùn)行直到被系統(tǒng)殺死。應(yīng)用程序流程是按需創(chuàng)建的,在您看到應(yīng)用程序的啟動活動啟動并運(yùn)行之前,發(fā)生了許多事情。
每個應(yīng)用程序都在其自己的進(jìn)程中運(yùn)行 :默認(rèn)情況下,每個Android應(yīng)用程序都在其自己的Android進(jìn)程中運(yùn)行,而這個進(jìn)程只不過是一個Linux進(jìn)程,而該進(jìn)程首先需要一個執(zhí)行線程。例如,當(dāng)您單擊電子郵件中的超鏈接時,網(wǎng)頁將在瀏覽器窗口中打開。您的郵件客戶端和瀏覽器是兩個單獨的應(yīng)用程序,它們分別在兩個單獨的進(jìn)程中運(yùn)行。click事件使Android平臺啟動新進(jìn)程,以便它可以在其自身進(jìn)程的上下文中實例化瀏覽器活動。這對于應(yīng)用程序中的任何其他組件同樣適用。
讓我們退后一會兒,快速瀏覽一下系統(tǒng)啟動過程。與大多數(shù)基于Linux的系統(tǒng)一樣,啟動加載程序在啟動時將加載內(nèi)核并啟動init進(jìn)程。然后,init會生成稱為“守護(hù)程序”的低級Linux進(jìn)程,例如android debug守護(hù)程序,USB守護(hù)程序等。這些守護(hù)程序通常處理低級硬件接口,包括無線電接口。
然后,初始化過程會啟動一個非常有趣的過程,稱為“zygote'。
顧名思義,這是其余Android應(yīng)用程序的開始。這是初始化Dalvik虛擬機(jī)的第一個實例的過程。它還預(yù)加載Android應(yīng)用程序框架和系統(tǒng)上安裝的各種應(yīng)用程序使用的所有常見類。因此,它準(zhǔn)備進(jìn)行復(fù)制。它統(tǒng)計偵聽套接字接口上的將來請求,以產(chǎn)生新的虛擬機(jī)(VM)來管理新的應(yīng)用程序進(jìn)程。收到新請求后,它會分叉以創(chuàng)建一個新進(jìn)程,該進(jìn)程將獲取預(yù)先初始化的VM實例。
zygote之后,init啟動運(yùn)行時過程。
然后zygote分叉以啟動一個名為System server的托管良好的進(jìn)程。系統(tǒng)服務(wù)器在其自己的上下文中啟動所有核心平臺服務(wù),例如活動管理器服務(wù)和硬件服務(wù)。
此時,完整的堆棧已準(zhǔn)備就緒,可以啟動第一個應(yīng)用程序流程-主頁應(yīng)用程序,該應(yīng)用程序顯示主屏幕(也稱為啟動器應(yīng)用程序)。
click事件被轉(zhuǎn)換為 startActivity(intent), 并通過Binder IPC路由到 ActivityManagerService 。ActvityManagerService執(zhí)行多個步驟
如您所見,當(dāng)用戶單擊圖標(biāo)并啟動新應(yīng)用程序時,許多事情發(fā)生在幕后。這是全圖:
流程創(chuàng)建:
ActivityManagerService 通過調(diào)用 startProcessLocked() 方法創(chuàng)建一個新進(jìn)程,該方法通過套接字連接將參數(shù)發(fā)送到Zygote進(jìn)程。Zygote派生自己并調(diào)用 ZygoteInit.main() ,然后實例化 ActivityThread 對象并返回新創(chuàng)建的進(jìn)程的進(jìn)程ID。
默認(rèn)情況下,每個進(jìn)程都有一個線程。主線程有一個 Looper 實例來處理來自消息隊列的消息,并且它在 run() 方法的每次迭代中都調(diào)用 Looper.loop() 。 Looper 的工作是從消息隊列中彈出消息并調(diào)用相應(yīng)的方法來處理它們。然后,ActivityThread通過隨后調(diào)用 Looper.prepareLoop() 和 Looper.loop()來 啟動消息循環(huán)。
以下序列詳細(xì)捕獲了調(diào)用序列:
figcaption class="jv jw di dg dh jx jy bo b fc cp ga" data-selectable-paragraph="" style="box-sizing: inherit; font-weight: 400; font-family: sohne, "Helvetica Neue", Helvetica, Arial, sans-serif; line-height: 20px; margin-left: auto; margin-right: auto; max-width: 728px; font-size: 14px; color: rgb(117, 117, 117); margin-top: 10px; text-align: center;"Android應(yīng)用啟動:單擊事件以執(zhí)行Looper調(diào)用順序/figcaption
應(yīng)用程序綁定:
下一步是將此新創(chuàng)建的過程附加到特定應(yīng)用程序。這是通過在線程對象上調(diào)用 bindApplication() 來完成的。此方法將 BIND_APPLICATION 消息發(fā)送到消息隊列。該消息由 Handler 對象檢索,該對象隨后調(diào)用 handleMessage() 方法以觸發(fā)特定于消息的操作 -handleBindApplication() 。此方法調(diào)用 makeApplication() 方法,該方法將應(yīng)用程序特定的類加載到內(nèi)存中。
下圖描述了該調(diào)用序列。
figcaption class="jv jw di dg dh jx jy bo b fc cp ga" data-selectable-paragraph="" style="box-sizing: inherit; font-weight: 400; font-family: sohne, "Helvetica Neue", Helvetica, Arial, sans-serif; line-height: 20px; margin-left: auto; margin-right: auto; max-width: 728px; font-size: 14px; color: rgb(117, 117, 117); margin-top: 10px; text-align: center;"Android應(yīng)用啟動:BIND_APPLICATION消息處理/figcaption
啟動活動:
在上一步之后,系統(tǒng)包含負(fù)責(zé)應(yīng)用程序的進(jìn)程,并將應(yīng)用程序類加載到進(jìn)程的私有內(nèi)存中。在新創(chuàng)建的流程和現(xiàn)有流程之間,啟動活動的調(diào)用順序很常見。
實際的啟動過程從 realStartActivity() 方法開始, 該 方法在應(yīng)用程序線程對象上調(diào)用 sheduleLaunchActivity() 。此方法將 LAUNCH_ACTIVITY 消息發(fā)送到消息隊列。該消息由 handleLaunchActivity() 方法處理,如下所示。
假設(shè)用戶單擊“視頻瀏覽器”應(yīng)用程序。啟動該活動的調(diào)用順序如圖所示。
figcaption class="jv jw di dg dh jx jy bo b fc cp ga" data-selectable-paragraph="" style="box-sizing: inherit; font-weight: 400; font-family: sohne, "Helvetica Neue", Helvetica, Arial, sans-serif; line-height: 20px; margin-left: auto; margin-right: auto; max-width: 728px; font-size: 14px; color: rgb(117, 117, 117); margin-top: 10px; text-align: center;"Android應(yīng)用啟動: LAUNCH_ACTIVITY消息處理 /figcaption
該活動通過 onCreate() 方法調(diào)用開始其托管生命周期。該活動通過 onRestart() 調(diào)用進(jìn)入前臺,并通過 onStart() 調(diào)用開始與用戶進(jìn)行交互。