作為Android開發者 你知道Android按下開機鍵到啟動發生什麼嗎?

前言

在一個夜黑風高的晚上,我的男同事突然給我發了一條微信,我點開來看,他竟然問我Android從按下開機鍵到啟動到底發生了什麼?此刻我的內心如下圖:


作為Android開發者 你知道Android按下開機鍵到啟動發生什麼嗎?


但是作為一個Android開發者,瞭解整個系統架構是必須的,所以這篇就總結一下Android手機從按下開機鍵到啟動這一過程發生了什麼。

要了解Android手機啟動過程,我們先來了解一下基於linux系統的電腦從按下電源鍵的那一刻起,發生了什麼,這樣類比可以更好的理解Android手機的啟動過程。

基於Linux的pc啟動過程

我們都知道,所有的程序軟件包括操作系統都是運行在內存中的,然而我們的操作系統一般是存放在硬盤上的,當我們按下開機鍵的時候,此時內存中什麼程序也沒有,因此需要藉助某種方式,將操作系統加載到內存中,而完成這項任務的就是 BIOS

裝過系統的人一定知道BIOS這個東西,那麼它究竟是什麼呢?

BIOS:Basic Input/Output System(基本輸入輸出系統),在IBM PC兼容系統上,是一種業界標準的固件接口(來自維基百科)。有點難以理解,其實BIOS是我們電腦啟動時加載的第一個程序,這個程序不是由Java語言編寫也不是由C語言編寫,一般是彙編程序。

BIOS程序固化在主板上的一塊芯片上,是連接計算機硬件與操作系統的橋樑,它保存著計算機最重要的基本輸入輸出的程序、開機後自檢程序和系統自啟動程序。

那麼問題來了,BIOS程序又是怎麼啟動的?BIOS的啟動,是由硬件完成的,Intel 80x86系列的cpu的硬件都設計為加電(即開機瞬間)就進入16位實模式狀態運行,此時將cpu的硬件邏輯設計為強行將CS的值設置為0xFFFF,IP的值設置為0x0000,這樣CS:IP就指向了0xFFFF0這個位置,而這個位置就是BIOS程序的入口地址。

因此這是一個硬件廠商之間的約定,所有的BIOS程序入口地址均為0xFFFF0,這樣在開機的時候,就找到這個地址,如果該地址並沒有代碼段,那麼計算機將會死機,如果這個地址處有代碼段,將會執行這個代碼段,並由此執行下去,即BIOS程序開始啟動。

補充:

CS:代碼段寄存器,存在於CPU中,指向CPU當前執行代碼在內存中所在的區域。

IP:指令寄存器,存在於CPU中,記錄將要執行的指令在代碼段內的偏移地址,與CS組合即為將要執行的指令的內存地址。

當BIOS程序啟動時,就會檢測硬件設備,比如我們的顯卡、內存等信息。BIOS會在內存中建立中斷向量表和中斷服務程序。中斷向量表中有256箇中斷向量,每個中斷向量佔4個字節,每個中斷向量指向一箇中斷服務程序,這些中斷服務程序完成了將操作系統由硬盤加載到內存中的任務.

基於linux的操作系統而言,計算機將分三批逐次加載操作系統的代碼,第一批由BIOS中斷int 0x19將 第一扇區bootsect的內容加載到內存;第二批和第三批在bootsect的指揮下,分別加載後面扇區的內容到內存中。

經過執行一系列的BIOS代碼後,計算機完成了自檢等操作,計算機硬件體系會與BIOS聯合操作,讓cpu接收到一個int 0x19中斷,cpu接收到這個中斷後,會立即在中斷向量表中找到int 0x19中斷向量,此時會找到對應的中斷服務程序,並由該中斷服務程序將硬盤中第一個扇區的引導程序加在到內存中的指定位置。

隨後,在引導程序的作用下,陸續將操作系統的其他程序載入內存,完成實模式到保護模式的轉變,為執行操作系統的入口函數main做準備,後面就是操作系統的初始化工作了,最後完成計算機的啟動。

Android手機的啟動過程

Android系統雖然也是基於linux系統的,但是由於Android屬於嵌入式設備,並沒有像pc那樣的BIOS程序。

取而代之的是Bootloader ——系統啟動加載器。它類似於BIOS,在系統加載前,用以初始化硬件設備,建立內存空間的映像圖,為最終調用系統內核準備好環境。

在Android裡沒有硬盤,而是ROM ,它類似於硬盤存放操作系統,用戶程序等。ROM跟硬盤一樣也會劃分為不同的區域,用於放置不同的程序,在Android中主要劃分為一下幾個分區:

  • /boot:存放引導程序,包括內核和內存操作程序
  • /system:相當於電腦c盤,存放Android系統及系統應用
  • /recovery:恢復分區,可以進入該分區進行系統恢復
  • /data:用戶數據區,包含了用戶的數據:聯繫人、短信、設置、用戶安裝的程序
  • /cache:安卓系統緩存區,保存系統最常訪問的數據和應用程序
  • /misc:包含一些雜項內容,如系統設置和系統功能啟用禁用設置
  • /sdcard:用戶自己的存儲區,可以存放照片,音樂,視頻等文件

那麼Bootloader是如何被加載的呢?我們可以想到,應該跟pc一樣,當開機加電的時候,cpu會從cpu製造廠商預設的地址上取指令,這個地址是各廠商約定俗稱的,類似於上面80x86架構裡的0xFFFF0地址,因此Android手機會將固態存儲設備ROM預先映射到該地址上,當開機加電的時候,cpu就會從該地址執行/boot分區下的Bootloader程序,載入linux內核到RAM中。

當linux內核啟動後會初始化各種軟硬件環境,加載驅動程序,掛載根文件系統,並開始執行根文件系統的init程序,init程序是Android啟動過程中最重要的核心程序。

init進程是Android系統中用戶進程的鼻祖進程。init進程會啟動各種系統本地服務,如:Media Server、Service Manager、bootanim(開機動畫)等。init進程會在解析init.rc文件後fork出Zygote,而Zygote是所有Java進程的父進程,我們的App都是由Zygote fork出來的。

Zygote進程主要包含:

  • 加載ZygoteInit類,註冊Zygote Socket服務端套接字;
  • 加載虛擬機;
  • 預加載Android核心類
  • 預加載系統資源

隨後Zygote進程會fork出System Server進程,System Server進程負責啟動和管理整個framework,包括Activity Manager,PowerManager等服務。

當System Server將系統服務啟動就緒後,就會通知ActivityManager啟動首個Android程序Home即我們看到的桌面程序。

至此,從Android手機開機到看到桌面程序所有過程分析完了。

最後附一張整體流程圖,幫助更好理解:


作為Android開發者 你知道Android按下開機鍵到啟動發生什麼嗎?

本人14年java轉Android開發,去阿里,華為等大廠待過,也面試過很多人。不少人私下問我,2019年Android進階該怎麼學,方法有沒有?深知大多數初中級Android工程師,想要提升技能,往往是自己摸索成長,不成體系的學習效果低效漫長且無助。

沒錯,三月份開始我花了一個多月的時間整理出來的學習資料,希望能幫助那些想進階提升Android開發,卻又不知道怎麼進階學習的朋友。【包括高級UI、性能優化、架構師課程、NDK、Kotlin、混合式開發(ReactNative+Weex)、Flutter等架構技術資料】,希望能幫助到您面試前的複習且找到一個好的工作,也節省大家在網上搜索資料的時間來學習。

資料獲取方式:轉發+關注後私信回覆我【學習】免費獲取Android進階開發資料!!!

相關推薦

推薦中...