這一篇主要是學(xué)習(xí)Android apk 的編譯流程
但其實(shí)谒亦,本人還是沒有成功編譯成一個(gè)apk的(或則是說鹿响,編譯成功了周伦,但運(yùn)行不起來)郑现,這里是用來做下筆記湃崩,方便以后可以回來再接著研究
環(huán)境:
1.deepin linux 15.8桌面版
2.android 27
3.java 1.8.0_171
不行的原因猜測:
1.在生成R文件的時(shí)候,我看網(wǎng)上很多文章都是寫只要工程的res文件接箫,但其實(shí)不然攒读,如果只有工程的res,你會發(fā)現(xiàn)只有一點(diǎn)辛友,肯定是不對的薄扁,對比下as生成的R文件對比就知道了。
? 所以废累,我把邓梅,library下的所有庫的res,都加進(jìn)來。這樣R文件跟as生成差不多了邑滨,但還是有些不同
2.生成class文件日缨,同理,網(wǎng)上文章也是只需要 項(xiàng)目下的 java文件掖看,但是會直接報(bào)錯(cuò)匣距,找不到v7某些類,所以哎壳,同樣把library相關(guān)的jar文件也引用來
3.生成dex文件毅待,同樣跟上面一樣
? 上面這上個(gè)地方,生成的R文件归榕,resources.arsc尸红,class,dex,跟as app生成的都不太一樣
4.META-INF是我從原apk拿出來的
本人找了不少思路去找哪里的東西不一致,但牽扯的文件太多驶乾,文件對比也找不出來邑飒。故思路卡住了循签,如果有相關(guān)經(jīng)驗(yàn)的级乐,希望能請教下
apk的編譯流程,網(wǎng)上有很多流程圖县匠,主要分下面步驟(這下面是本人實(shí)驗(yàn)出來的风科,不一定對的,因?yàn)楦W(wǎng)上大部分文章都不一樣)
1.通過 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拿過來的)放進(jìn)文件夾里玖姑,再通過壓縮愕秫,改后綴名成apk
6.簽名,對齊焰络,安裝
1.通過 aapt 生成R文件和資源文件resources.arsc
需要:工程的res戴甩,引用庫的res(在as左側(cè) External Libraries 打開,資源路徑指向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.通過aidl 生成java
需要:framework.aidl? 項(xiàng)目aidl
./aidl -I$projectPath/aidl -p$platformsPath/framework.aidl -o$workPath/aidl $projectPath/aidl/$apkName/IPlusService.aidl
3.通過javac 生成class
需要:android.jar ,項(xiàng)目java ,aidl生成的java,引用庫的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.通過dx 生成dex
需要:項(xiàng)目java的class,引用庫的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本來有個(gè)生成apk工具,但不知道從那個(gè)版本開始描馅,已經(jīng)沒了把夸,其實(shí)最簡單的方式,就是寫個(gè)腳本流昏,把需要的文件壓縮扎即,改名成apk就可以了。親測可以用
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.對齊
? ? ./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è)步驟的分開的步驟的代碼(以buildApk為準(zhǔn)况凉,其他的有部分在調(diào)試的時(shí)候有些修改了代碼)
注意:
1.簽名文件谚鄙,記得自己提供
2.項(xiàng)目最好用一個(gè)簡單的項(xiàng)目,一個(gè)只有一個(gè)hellowork的最好
3.library需要替換成自己的
4.最好刁绒,一個(gè)步驟分開來跑闷营,雖然代碼我測過,但其他環(huán)境能不能跑起來,就不確定了
我的報(bào)錯(cuò):
java.lang.NoClassDefFoundError: Failed resolution of: Landroid/support/v7/appcompat/R$drawable;
擴(kuò)散思考:
1.為什么傻盟,apk 運(yùn)行不了速蕊,覺得還是dex那里出了問題,因?yàn)槲沂菆?bào)找不到某個(gè)dex
2.我現(xiàn)在用的是aapt娘赴,其實(shí)已經(jīng)有aapt2了规哲,網(wǎng)上大部分文章都是基于這個(gè)寫的,粗略查了下诽表,aapt是全量的唉锌,但aapt2是增量的,對應(yīng)gradle里面的Instant Run 竿奏“兰颍可以吃一下,鉤上這個(gè)泛啸,還不鉤這個(gè)绿语,打出來的apk里面的文件是不同的
3.如果用kotlin,是不是打包流程不一樣候址,還是多一個(gè)把kotlin賺class而已
4.學(xué)習(xí)的時(shí)候吕粹,找到另一個(gè)知識點(diǎn),好像從某個(gè)版本開始(好像5.0)系統(tǒng)就不用dex(DVM),而是用 ART了宗雇,那對這流程有什么不同呢昂芜?
5.在apk安裝的時(shí)候,dex被系統(tǒng)優(yōu)化過赔蒲,做了什么呢泌神?好像如果是art也做了類似的步驟
demo:
https://github.com/raqusty/Pluggable
在這個(gè)github的workPlace下面,注意舞虱,路徑有所變化欢际,keystore文件自己加上去
最后,希望這文章對你有所幫助矾兜,有錯(cuò)請指出损趋,萌新一枚。如果最后你成功把a(bǔ)pk跑起來了椅寺,麻煩告知一聲浑槽,拜謝了~