Dalvik虛擬機

一、什麼是Dalvik?

Google於2007年底正式發佈了Android SDK, 作爲 Android系統的重要特性,Dalvik虛擬機也第一次進入了人們的視野。它對內存的高效使用,和在低速CPU上表現出的高性能,確實令人刮目相看。 依賴於底層Posix兼容的操作系統,它可以簡單的完成進程隔離和線程管理。每一個Android應用在底層都會對應一個獨立的Dalvik虛擬機實例, 其代碼在虛擬機的解釋下得以執行。

二、Dalvik與JVM的區別
很多人認爲Dalvik虛擬機是一個Java虛擬機,因爲Android的編程語言恰恰就是Java語言。但是這種說法並不準確,因爲Dalvik虛擬機並不是按照Java虛擬機的規範來實現的,兩者並不兼容;同時還要兩個明顯的不同:

  • Java虛擬機運行的是Java字節碼,而Dalvik虛擬機運行的則是其專有的文件格式DEX(Dalvik Executable)。

  • 在Java SE程序中的Java類會被編譯成一個或者多個字節碼文件(.class)然後打包到JAR文件,而後Java虛擬機會從相應的CLASS文件和JAR文 件中獲取相應的字節碼;Android應用雖然也是使用Java語言進行編程,但是在編譯成CLASS文件後,還會通過一個工具(dx)將應用所有的 CLASS文件轉換成一個DEX文件,而後Dalvik虛擬機會從其中讀取指令和數據。

Dalvik和Android系統Android作爲新一代的基於Linux的開源手機操作系統,其系統架構由下而上可以分爲以下幾部分:

  • Linux內核

  • 本地庫

  • Android運行庫

  • 應用框架

  • 應用

java虛擬機和Dalvik虛擬機的區別:


java虛擬機Dalvik虛擬機

java虛擬機基於。 基於棧的機器必須使用指令來載入和操作棧上數據,所需指令更多更多dalvik虛擬機是基於寄存器

java虛擬機運行的是java字節碼。(java類會被編譯成一個或多個字節碼.class文件,打包到.jar文件中,java虛擬機從相應的.class文件和.jar文件中獲取相應的字節碼)


Dalvik運行的是自定義的.dex字節碼格式。(java類被編譯成.class文件後,會通過一個dx工具將所有的.class文件轉換成一個.dex文件,然後dalvik虛擬機會從其中讀取指令和數據)


常量池已被修改爲只使用32位的索引,以 簡化解釋器。dalvik的堆和棧的參數可以通過-Xms和-Xmx更改


一個應用,一個虛擬機實例,一個進程(所有android應用的線程都是對應一個linux線程,都運行在自己的沙盒中,不同的應用在不同的進程中運行。每個android dalvik應用程序都被賦予了一個獨立的linux PID(app_*))


Dalvik和標準Java虛擬機(JVM)之間的首要差別之一,就是Dalvik基於寄存器,而JVM基於棧。
Dalvik和Java之間的另外一大區別就是運行環境——Dalvik經過優化,允許在有限的內存中同時運行多個虛擬機的實例,並且每一個 Dalvik應用作爲一個獨立的Linux進程執行。
(1)虛擬機很小,使用的空間也小;
(2)Dalvik沒有JIT編譯器;
(3)常量池已被修改爲只使用32位的索引,以簡化解釋器;
(4)它使用自己的字節碼,而非Java字節碼。

184139971.jpg


Dalvik虛擬機架構:

在android源碼中,Dalvik虛擬機的實現位於“dalvik/”目錄下,其中“dalvik/vm”是虛擬機的實現部分,將會編譯成libdvm.so;而"dalvik/libdex"將會編譯成libdex.a靜態庫作爲dex工具;“dalvik/dexdump”是.dex文件的反編譯工具;虛擬機的可執行程序位於“dalvik/dalvikvm”中,將會編譯成dalvikvm可執行文件。

dalvik虛擬機架構:

183741928.png

Android應用編譯及運行流程:

183847871.png

Dalvik進程管理:

dalvik進程管理是依賴於linux的進程體系結構的,如要爲應用程序創建一個進程,它會使用linux的fork機制來複制一個進程(複製進程往往比創建進程效率更高)。

Zygote是一個虛擬機進程,同時也是一個虛擬機實例的孵化器,它通過init進程啓動。首先會孵化出System_Server(android絕大多系統服務的守護進程,它會監聽socket等待請求命令,當有一個應用程序啓動時,就會向它發出請求,zygote就會FORK出一個新的應用程序進程).每當系統要求執行一個android應用程序時,Zygote就會運用linux的FORK進制產生一個子進程來執行該應用程序。


JVM和Dalvik進程管理:


linux中進程間通信的方式有很多,但是dalvik使用的是信號方式來完成進程間通信。


Android的初始化流程

183937505.png

三、Dalvik和Android系統

Android作爲新一代的基於Linux的開源手機操作系統,其系統架構由下而上可以分爲以下幾部分(Linux內核、本地庫、Android運行庫、應用框架、應用程序)

如圖所示:

103533224.png

Android運行庫包括兩部分:核心庫和Dalvik虛擬機。核心庫包括了最基本的類庫,如data structure,network, Utilities, File system等的,很多實現代碼都是來自Apache Harmony項目,主要目的是保證虛擬機的類庫能夠和Java SE的類庫最大可能的兼容,從而降低應用開發者從Java SE陣營轉移到Android開發陣營的難度,增加其可用性。Dalvik虛擬機主要是完成對象生命週期的管理,堆棧的管理,線程管理,安全和異常的管理,以及垃圾回收等等重要功能。

四、 Dalvik虛擬機的主要特徵

Dalvik虛擬機非常適合在移動終端上使用,相對於在桌面系統和服務器系統運行的虛擬機而言,它不需要很快的CPU速度和大量的內存空間。根據Google的測算,64M的RAM已經能夠令系統正常運轉了。其中24M被用於底層系統的初始化和啓動,另外20M被用於高層啓動高層服務。當然,隨着系統服務的增多和應用功能的擴展,其所消耗的內存也勢必越來越大。

歸納起來,Dalvik虛擬機有如下幾個主要特徵:

1) 專有的DEX文件格式

DEX是Dalvik虛擬機專用的文件格式,而爲什麼棄用已有的字節碼文件(CLASS文件)而採用新的格式呢?

1. 一個應用中會定義很多類,編譯完成後即會有很多相應的CLASS文件,CLASS文件間會有不少冗餘的信息;而DEX文件格 式會把所有的CLASS文件內容整合到一個文件中。這樣,除了減少整體的文件尺寸,I/O操作,也提高了類的查找速度。原來每個 類文件中的常量池,在DEX文件中由一個常量池來管理,具體方式如下圖:


101610565.jpg

101644518.jpg

2. 增加了新的操作碼的支持

3. 文件結構儘量簡潔,使用等長的指令,藉以提高解析速度

4. 儘量擴大隻讀結構的大小,藉以提高跨進程的數據共享

如何生成DEX文件呢?Android系統和Dalvik虛擬機提供了工具(DX),在把Java源代碼編譯成CLASS文件後,使用DX工具。

103218484.jpg

2) DEX的優化

DEX文件的結構是緊湊的,然是如果我們還想要求運行時的性能有進一步提高,我們就仍然需要對DEX文件進行進一步優化。優化主要是針對以下幾個方面:

1. 調整所有字段的字節序(LITTLE_ENDIAN)和對齊結構中的沒一個域

2. 驗證DEX文件中的所有類

3. 對一些特定的類進行優化,對方法裏的操作碼進行優化。優化後的文件大小會有所增加,應該是原DEX文件的1-4倍。優化發生的時機有兩個:對於預置應用,可以在 系統編譯後,生成優化文件,以ODEX結尾。這樣在發佈時除APK文件(不包含DEX)以外,還有一個相應的ODEX文件;對於非預置應用,包含在APK文件裏的DEX文件會 在運行時被優化,優化後的文件將被保存在緩存中。

3) 基於寄存器

相對於基於堆棧的虛擬機實現,基於寄存器的虛擬機實現雖然在硬件通用性上要差一些,但是它在代碼的執行效率上卻更勝一籌。一般來講,虛擬機中指令的解釋執行時間主要花在以下三個方面:

1. 分發指令

2. 訪問運算數

3. 執行運算

其中“分發指令”這個環節對性能的影響最大。在基於寄存器的虛擬機裏,可以更爲有效的減少冗餘指令的分發和減少內存的讀寫訪問,如:

\

雖然Dalvik虛擬機並沒有使用目前流行的虛擬機技術,如JIT,但是根據Google的報告,這個功能的缺失並沒有另Dalvik虛擬機在性能上有所損失。我們也同時相信,Dalvik虛擬機的性能還有進一步提高的空間。

4) 一個應用,一個虛擬機實例,一個進程

每一個Android應用都運行在一個Dalvik虛擬機實例裏,而每一個虛擬機實例都是一個獨立的進程空間。虛擬機的線程機制,內存分配和管理,Mutex等等都是依賴底層操作系統而實現的。所有Android應用的線程都對應一個Linux線程,虛擬機因而可以更多的依賴操作系統的線程調度和管理機制。

不同的應用在不同的進程空間裏運行,加之對不同來源的應用都使用不同的Linux用戶來運行,可以最大程度的保護應用的安全和獨立運行。

Zygote是一個虛擬機進程,同時也是一個虛擬機實例的孵化器,每當系統要求執行一個Android應用程序,Zygote就會FORK出一個子進程來執行該應用程序。這樣做的好處顯而易見:Zygote進程是在系統啓動時產生的,它會完成虛擬機的初始化,庫的加載,預置類庫的加載和初始化等等操作,而在系統需要一個新的虛擬機實例時,Zygote通過複製自身,最快速的提供個系統。另外,對於一些只讀的系統庫,所有虛擬機實例都和Zygote共享一塊內存區域,大大節省了內存開銷。


101417590.gif

應用程序包(APK)被髮布到手機上後,運行前會對其中的DEX文件進行優化,優化後的文件被保存到緩存區域(優化後的格式被稱爲DEY),虛擬機會直接執行該文件。如果應用包文件不發生變化,DEY文件不會被重新生成。

102215795.jpg

五、Android應用開發和Dalvik虛擬機

Android應用所使用的編程語言是Java語言,和Java SE一樣,編譯時使用Sun JDK將Java源程序編程成標準的Java字節碼文件(.class文件),而後通過工具軟件DX把所有的字節碼文件轉成DEX文件(classes.dex)。最後使用Android打包工具(aapt)將DEX文件,資源文件以及AndroidManifest.xml文件(二進制格式)組合成一個應用程序包(APK)。應用程序包可以被髮布到手機上運行。

102106255.jpg


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章