学习Android apk 的编译流程

2019-04-02  本文已影响0人  HayasiR

这一篇主要是学习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跑起来了,麻烦告知一声,拜谢了~

上一篇下一篇

猜你喜欢

热点阅读