这一篇主要是学习Android apk 的编译流程
但其实,本人还是没有成功编译成一个apk的(或则是说,编译成功了,但运行不起来),这里是用来做下笔记,方便以后可以回来再接着研究
环境:
1.deepin linux 15.8桌面版
2.android 27
3.java 1.8.0_171
不行的原因猜测:
1.在生成R文件的时候,我看网上很多文章都是写只要工程的res文件,但其实不然,如果只有工程的res,你会发现只有一点,肯定是不对的,对比下as生成的R文件对比就知道了。
所以,我把,library下的所有库的res,都加进来。这样R文件跟as生成差不多了,但还是有些不同
2.生成class文件,同理,网上文章也是只需要 项目下的 java文件,但是会直接报错,找不到v7某些类,所以,同样把library相关的jar文件也引用来
3.生成dex文件,同样跟上面一样
上面这上个地方,生成的R文件,resources.arsc,class,dex,跟as app生成的都不太一样
4.META-INF是我从原apk拿出来的
本人找了不少思路去找哪里的东西不一致,但牵扯的文件太多,文件对比也找不出来。故思路卡住了,如果有相关经验的,希望能请教下
apk的编译流程,网上有很多流程图,主要分下面步骤(这下面是本人实验出来的,不一定对的,因为跟网上大部分文章都不一样)
1.通过 aapt 生成R文件和资源文件resources.arsc(其实是一个压缩包,解压在里面)
2.aidl生成java文件,如果有aidl的话
3.将生成的aidl生成的java文件,和项目的java文件,生成class文件
4.将class生成dex文件
5.将第一部生成的压缩包解压,里面有res,resources.arsc,androidManifest.xml,再把classes.dex,和META-INF文件(目前是从原app拿过来的)放进文件夹里,再通过压缩,改后缀名成apk
6.签名,对齐,安装
1.通过 aapt 生成R文件和资源文件resources.arsc
需要:工程的res,引用库的res(在as左侧 External Libraries 打开,资源路径指向gradle的下载路径),Android.jar,项目的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 项目aidl
./aidl -I$projectPath/aidl -p$platformsPath/framework.aidl -o$workPath/aidl $projectPath/aidl/$apkName/IPlusService.aidl
3.通过javac 生成class
需要:android.jar ,项目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
需要:项目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本来有个生成apk工具,但不知道从那个版本开始,已经没了,其实最简单的方式,就是写个脚本,把需要的文件压缩,改名成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 是整个流程的sh
2.其他的是每个步骤的分开的步骤的代码(以buildApk为准,其他的有部分在调试的时候有些修改了代码)
注意:
1.签名文件,记得自己提供
2.项目最好用一个简单的项目,一个只有一个hellowork的最好
3.library需要替换成自己的
4.最好,一个步骤分开来跑,虽然代码我测过,但其他环境能不能跑起来,就不确定了
我的报错:
java.lang.NoClassDefFoundError: Failed resolution of: Landroid/support/v7/appcompat/R$drawable;
扩散思考:
1.为什么,apk 运行不了,觉得还是dex那里出了问题,因为我是报找不到某个dex
2.我现在用的是aapt,其实已经有aapt2了,网上大部分文章都是基于这个写的,粗略查了下,aapt是全量的,但aapt2是增量的,对应gradle里面的Instant Run 。可以吃一下,钩上这个,还不钩这个,打出来的apk里面的文件是不同的
3.如果用kotlin,是不是打包流程不一样,还是多一个把kotlin赚class而已
4.学习的时候,找到另一个知识点,好像从某个版本开始(好像5.0)系统就不用dex(DVM),而是用 ART了,那对这流程有什么不同呢?
5.在apk安装的时候,dex被系统优化过,做了什么呢?好像如果是art也做了类似的步骤
demo:
https://github.com/raqusty/Pluggable
在这个github的workPlace下面,注意,路径有所变化,keystore文件自己加上去
最后,希望这文章对你有所帮助,有错请指出,萌新一枚。如果最后你成功把apk跑起来了,麻烦告知一声,拜谢了~