這一篇主要是學(xué)習(xí)Android apk 的編譯流程
但其實(shí),本人還是沒(méi)有成功編譯成一個(gè)apk的(或則是說(shuō),編譯成功了,但運(yùn)行不起來(lái)),這里是用來(lái)做下筆記,方便以后可以回來(lái)再接著研究
環(huán)境:
1.deepin linux 15.8桌面版
2.android 27
3.java 1.8.0_171
不行的原因猜測(cè):
1.在生成R文件的時(shí)候,我看網(wǎng)上很多文章都是寫(xiě)只要工程的res文件,但其實(shí)不然,如果只有工程的res,你會(huì)發(fā)現(xiàn)只有一點(diǎn),肯定是不對(duì)的,對(duì)比下as生成的R文件對(duì)比就知道了。
? 所以,我把,library下的所有庫(kù)的res,都加進(jìn)來(lái)。這樣R文件跟as生成差不多了,但還是有些不同
2.生成class文件,同理,網(wǎng)上文章也是只需要 項(xiàng)目下的 java文件,但是會(huì)直接報(bào)錯(cuò),找不到v7某些類,所以,同樣把library相關(guān)的jar文件也引用來(lái)
3.生成dex文件,同樣跟上面一樣
? 上面這上個(gè)地方,生成的R文件,resources.arsc,class,dex,跟as app生成的都不太一樣
4.META-INF是我從原apk拿出來(lái)的
本人找了不少思路去找哪里的東西不一致,但牽扯的文件太多,文件對(duì)比也找不出來(lái)。故思路卡住了,如果有相關(guān)經(jīng)驗(yàn)的,希望能請(qǐng)教下
apk的編譯流程,網(wǎng)上有很多流程圖,主要分下面步驟(這下面是本人實(shí)驗(yàn)出來(lái)的,不一定對(duì)的,因?yàn)楦W(wǎng)上大部分文章都不一樣)
1.通過(guò) aapt 生成R文件和資源文件resources.arsc(其實(shí)是一個(gè)壓縮包,解壓在里面)
2.aidl生成java文件,如果有aidl的話
3.將生成的aidl生成的java文件,和項(xiàng)目的java文件,生成class文件
4.將class生成dex文件
5.將第一部生成的壓縮包解壓,里面有res,resources.arsc,androidManifest.xml,再把classes.dex,和META-INF文件(目前是從原app拿過(guò)來(lái)的)放進(jìn)文件夾里,再通過(guò)壓縮,改后綴名成apk
6.簽名,對(duì)齊,安裝
1.通過(guò) aapt 生成R文件和資源文件resources.arsc
需要:工程的res,引用庫(kù)的res(在as左側(cè) External Libraries 打開(kāi),資源路徑指向gradle的下載路徑),Android.jar,項(xiàng)目的androidManifest.xml,R文件輸出路徑,資源文件輸出文件名
./aapt? package --auto-add-overlay -f -m -S $projectPath/res/ -S $resName5 -S $resName11? -I $platformsPath/android.jar -M $projectPath/AndroidManifest.xml -J $workPath/ -F $workPath/resources/myApk.zip
如果不理解,輸入 ./aapt --help
2.通過(guò)aidl 生成java
需要:framework.aidl? 項(xiàng)目aidl
./aidl -I$projectPath/aidl -p$platformsPath/framework.aidl -o$workPath/aidl $projectPath/aidl/$apkName/IPlusService.aidl
3.通過(guò)javac 生成class
需要:android.jar ,項(xiàng)目java ,aidl生成的java,引用庫(kù)的jar
javac -encoding utf-8 -target 1.8 -d $workPath/class $workPath/aidl/$apkName/*.java $projectPath/java/$apkName/*.java $workPath/$apkName/R.java -bootclasspath $platformsPath/android.jar:$jarName1:$jarName2:$jarName3:$jarName4:$jarName5:$jarName6:$jarName7:$jarName8:$jarName9:$jarName10:$jarName11:$jarName12:$jarName13:$jarName14:$jarName15:$jarName16:$jarName17:$jarName18:$jarName19:$jarName20:$jarName21:$jarName22:$jarName23:$jarName26:$jarName27:$jarName28:$jarName24:$jarName25
4.通過(guò)dx 生成dex
需要:項(xiàng)目java的class,引用庫(kù)的jar
./dx --dex --output=$workPath/dex/classes.dex? $workPath/class? $jarName1 $jarName2 $jarName3 $jarName4 $jarName5 $jarName6 $jarName7 $jarName8 $jarName9 $jarName10 $jarName11 $jarName12 $jarName13 $jarName14 $jarName15 $jarName16 $jarName17 $jarName18 $jarName19 $jarName20 $jarName21 $jarName22 $jarName23 $jarName26 $jarName28 $jarName27 $jarName24 $jarName25
5.合并apk
sdk本來(lái)有個(gè)生成apk工具,但不知道從那個(gè)版本開(kāi)始,已經(jīng)沒(méi)了,其實(shí)最簡(jiǎn)單的方式,就是寫(xiě)個(gè)腳本,把需要的文件壓縮,改名成apk就可以了。親測(cè)可以用
6.簽名
./apksigner sign? --ks $workPath/keystore/xxx.jks? --ks-key-alias xxx? --ks-pass pass:xxx? --key-pass pass:xxx? --out $workPath/output.apk? $workPath/my.apk
7.對(duì)齊
? ? ./zipalign -v -p 4 $workPath/output.apk $workPath/finish.apk
8.安裝
adb? install -r -t $workPath/finish.apk
下面是用到的shell文件
1.buildApk 是整個(gè)流程的sh
2.其他的是每個(gè)步驟的分開(kāi)的步驟的代碼(以buildApk為準(zhǔn),其他的有部分在調(diào)試的時(shí)候有些修改了代碼)
注意:
1.簽名文件,記得自己提供
2.項(xiàng)目最好用一個(gè)簡(jiǎn)單的項(xiàng)目,一個(gè)只有一個(gè)hellowork的最好
3.library需要替換成自己的
4.最好,一個(gè)步驟分開(kāi)來(lái)跑,雖然代碼我測(cè)過(guò),但其他環(huán)境能不能跑起來(lái),就不確定了
我的報(bào)錯(cuò):
java.lang.NoClassDefFoundError: Failed resolution of: Landroid/support/v7/appcompat/R$drawable;
擴(kuò)散思考:
1.為什么,apk 運(yùn)行不了,覺(jué)得還是dex那里出了問(wèn)題,因?yàn)槲沂菆?bào)找不到某個(gè)dex
2.我現(xiàn)在用的是aapt,其實(shí)已經(jīng)有aapt2了,網(wǎng)上大部分文章都是基于這個(gè)寫(xiě)的,粗略查了下,aapt是全量的,但aapt2是增量的,對(duì)應(yīng)gradle里面的Instant Run 。可以吃一下,鉤上這個(gè),還不鉤這個(gè),打出來(lái)的apk里面的文件是不同的
3.如果用kotlin,是不是打包流程不一樣,還是多一個(gè)把kotlin賺class而已
4.學(xué)習(xí)的時(shí)候,找到另一個(gè)知識(shí)點(diǎn),好像從某個(gè)版本開(kāi)始(好像5.0)系統(tǒng)就不用dex(DVM),而是用 ART了,那對(duì)這流程有什么不同呢?
5.在apk安裝的時(shí)候,dex被系統(tǒng)優(yōu)化過(guò),做了什么呢?好像如果是art也做了類似的步驟
demo:
https://github.com/raqusty/Pluggable
在這個(gè)github的workPlace下面,注意,路徑有所變化,keystore文件自己加上去
最后,希望這文章對(duì)你有所幫助,有錯(cuò)請(qǐng)指出,萌新一枚。如果最后你成功把a(bǔ)pk跑起來(lái)了,麻煩告知一聲,拜謝了~