1. APK簡(jiǎn)介:
APK其實(shí)就是一個(gè)壓縮包,它里面包括:
- classes.dex:Dex是Android平臺(tái)上的可執(zhí)行文件,Android虛擬機(jī)Dalvik/Art支持的字節(jié)碼文件格式。通過(guò)Android平臺(tái)上的工具可以將Java字節(jié)碼轉(zhuǎn)換為Dex字節(jié)碼,轉(zhuǎn)換時(shí)會(huì)壓縮常量池,消除冗余信息等。同時(shí).class是基于棧而.dex是基于寄存器的,更適合移動(dòng)端。
- res:資源文件夾
- resources.arsc:這個(gè)文件記錄了所有應(yīng)用程序資源目錄的信息,包括每一個(gè)資源名稱(chēng),類(lèi)型,值,ID以及所配置的維度信息。resources.arsc是資源索引表,可以在給定資源ID和設(shè)備配置信息的情況下,能夠在應(yīng)用程序目錄中快速找到最匹配的資源。
- lib:存放so動(dòng)態(tài)鏈接庫(kù)。
- META-INF:簽名文件夾,存放了3個(gè)文件。2個(gè)是對(duì)資源文件進(jìn)行SHA1處理,一個(gè)是簽名和公鑰證書(shū)
- AndroidManifest.xml:是每個(gè)應(yīng)用程序必須的文件,描述了package中的組件,如Activity,service等,及它們的啟動(dòng)模式等。還能制定permissions(權(quán)限控制)和instrumentation(測(cè)試)
2. 打包流程:
-
使用aapt打包資源文件,生成R.java文件:
首先檢查AndroidManifest.xml的合法性,然后對(duì)res目錄下的資源目錄進(jìn)行處理。處理內(nèi)容包括文件名的合法性檢查,向資源表添加條目等。
-
編譯res與asserts目錄下的資源生成resource.arsc文件。然后生成R.java文件。
resource.arsc:清單文件,是一個(gè)資源索引表,可以方便地讓?xiě)?yīng)用程序運(yùn)行時(shí)根據(jù)ID來(lái)找到不同的設(shè)備分辨率對(duì)應(yīng)的資源。主要面向程序。
R.java:appt工具對(duì)每個(gè)資源文件都生成了唯一ID(除了assets外),這些ID保存在R.java中。主要面向程序員。
-
編譯res目錄下的xml,編譯過(guò)的xml文件就簡(jiǎn)單的被加密。最后將所有資源與編譯生成的resource.arsc文件,以及加密AndroidManifest.xml打包壓縮成resources.ap_文件。除了assets和res/raw資源被原封不懂地打包進(jìn)APK之外,其他資源都會(huì)被編譯或者處理。
將xml編譯成二進(jìn)制文件,占用空間會(huì)更小。因?yàn)樗械膞ml元素的標(biāo)簽,屬性名稱(chēng),屬性值和內(nèi)容所涉及到的字符串都會(huì)被統(tǒng)一收集到一個(gè)字符串資源池中并去重。有了這個(gè)字符串資源池,使用字符串的地方就會(huì)被替換成一個(gè)整數(shù)索引值,從而減小文件的大小。同時(shí)由于二進(jìn)制的XML元素里不再包含字符串值,避免了字符串解析,速度也會(huì)提高。
aapt傳統(tǒng)的打包主要是指res和Java代碼的打包,aapt打包走的是單線(xiàn)程。傳統(tǒng)的aapt會(huì)執(zhí)行2次,第一次是生產(chǎn)R.java,參與javac編譯,第二次是對(duì)res里面的文件進(jìn)行編譯。最后將Dex文件與編譯好的資源文件打包成apk,進(jìn)行簽名。整個(gè)流程沒(méi)有任務(wù)緩存,沒(méi)有并發(fā),沒(méi)有增量,每次構(gòu)建都是一個(gè)全新的流程。因此每次構(gòu)建時(shí)間都比較恒定,代碼量,資源量越多,構(gòu)建的時(shí)間就越慢。
使用AIDL工具處理AIDL文件,生成對(duì)應(yīng)的java文件
編譯工程源碼,生成相應(yīng)的class文件:使用javac工具編譯src目錄下的所有源文件,生成class文件。這些源文件包括R.java,AIDL生成的java文件,庫(kù)jar文件。
生成dex文件:使用dx工具將class文件轉(zhuǎn)換為dex文件。轉(zhuǎn)換時(shí)會(huì)壓縮常量池,消除冗余信息等。
打包生成apk:使用sdklib的ApkBuilderMain將打包后的資源文件,dex文件,lib文件打包成APK
對(duì)apk進(jìn)行簽名:Android的應(yīng)用程序需要簽名才能安裝。調(diào)試時(shí)會(huì)默認(rèn)使用debug.keystore對(duì)apk進(jìn)行簽名
-
對(duì)簽名后的apk進(jìn)行簽名處理:使用zipalign工具對(duì)apk進(jìn)行對(duì)齊處理
對(duì)齊處理主要是將文件的起始位置偏移為4字節(jié)的整數(shù)倍,這樣通過(guò)內(nèi)存映射訪(fǎng)問(wèn)apk時(shí)的速度會(huì)更快。
因?yàn)槿绻總€(gè)資源的開(kāi)始位置都是上一個(gè)資源之后的4n字節(jié)。那么訪(fǎng)問(wèn)下一個(gè)資源就不用遍歷,直接跳到4n字節(jié)處即可。