理解Android编译过程

Github链接

Androd运行环境

标准Java编译器使用JIT技术(见注1)将源代码编译成字节码(.class文件),紧接着进行第二轮编译(是对字节码进行压缩)出的字节码可以在特定的JVM运行。对于Android会有两个不同的虚拟机:

  • DALVIK:对于Android 4.4及以下版本使用的虚拟机,它使用JIT编译技术,但是在应用打开时进行编译,将应用启动速度变慢。
  • ART:应用AOT(ahead-of-time)技术在应用安装时将一部分的字节码转化成机器码,因此应用的启动速度变快。同时ART对GC技术进行了优化,默认采用CMS(并发标记和删除,见注2)缩短用户等待的时间。ART也存在缺点:会使安装速度变慢,因为缓存了机器码导致应用占用空间变大。

Android编译过程

java文件简要编译流程.png

更详细的编译补充过程见注3。

编译字节码

Android应用需要JDK来编译Java文件,通过javac命令来编译我们所写的代码,包括运行时的预编译类和三方库,输出字节码。这个步骤是一般的Java步骤,接下来才是Android特色的东西。

混淆(ProGuard Tool)

这个步骤是选择性开启的。Proguard Tool获取上述步骤输出的字节码,将其压缩(移除没人调用的方法)和混淆(将类名映射为无意义的名字)。

混淆大量减少包名,同时让破解者的成本大大提升。

不管用不用Proguard,这个步骤最后输出的也是.class文件(使用了就是混淆过的.class文件)。

.class字节码转化dex字节码

取到上一步输出的字节码将其转化为dex字节码(Dalvik Executable)。dex字节码是经过优化后的代码,能在Dalvik和ART运行时执行。

打包Apk并签名

将上一步输出的dex字节码和资源文件通过aapt一并压缩到apk内,通过jarsigner(标准签名工具)或zipalign(将字节码对齐)等工具进行签名。

运行apk

Zygote进程负责启动app。Zygote进程包含了所有应用公用的核心仓库,使用linux系统的fork命令来创建(非常快)一个自己的复制进程用于运行apk,这种方法会比启动新一个新进程再加载系统文件快得多。

执行机器码

Android运行时会读取上一步输出apk中的dex字节码,为了获得更快的执行速度,需要重新编译成机器码,可以成为ELF文件,在Android平台上是OAT文件。OAT文件会通过页映射到进程的内存里,然后把CPU执行入口地址改为该进程的内存地址开始执行(见注4),具体可以参考《程序员的自我修养—链接、装载与库》。

在Dalvki上这步在应用启动时使用JIT技术实现;

而在ART上,这步发生在app安装时,所以当启动应用时,速度会更快。

到这里位置,应用就启动主Activity并显示到屏幕上,具体可参考应用启动流程。

总结

从编写Java代码开始,经过一系列转换和编译,最终生成Android能执行的机器码,这个中间发生的东西还有很多,这里给了个大体流程。

1.JIT

Android 运行时 (ART) 包含一个具备代码分析功能的即时 (JIT) 编译器,该编译器可以在 Android 应用运行时持续提高其性能。JIT 编译器补充了 ART 当前的预先 (AOT) 编译器的功能,有助于提高运行时性能,节省存储空间,以及加快应用及系统更新速度。相较于 AOT 编译器,JIT 编译器的优势也更为明显,因为它不会在应用自动更新期间或重新编译应用(在无线下载 (OTA) 更新期间)时拖慢系统速度。

  • 用户打开应用,会另ART加载.dex文件:如果有.oat文件(.dex文件的AOT二进制文件),ART则会使用该文件;若果没有则使用JIT解释执行.dex文件
  • 针对未编译的应用启动JIT
  • 将JIT数据转存在只限应用访问的目录
  • AOT编译守护进程通过解析该文件来推动编译。

图片来自Android官方文档


jit-workflow.png

2.CMS

CMS是以获取最短回收停顿时间为目标的收集器。整个过程分为初始标记,并发标记,重新标记和并发清除,其中1和3都需要阻塞主线程来标记GC Root能关联的对象,而并发标记和并发清除是耗时最长的步骤。其缺点有:较为占用CPU,无法清除浮动垃圾和无法整理空间碎片。

3.构建流程

图片来自Android官方文档


构建流程.png

4.装载

图片来自《程序员的自我修养—链接、装载与库》


页错误.png

左边的ELF文件就是Android下执行的OAT文件,它按读写权限划分,具体分为.text代码段,.data数据段;中间是进程的虚拟地址空间,在Android下是Zygote fork出来的进程;右边是物理内存空间。

整个流程大概是,创建进程的虚拟地址空间并映射物理空间,建立ELF文件与虚拟空间的映射关系,开始执行。

具体请参考《程序员的自我修养—链接、装载与库》。

参考资料

Understanding the Android CompilationProcess -- By Bhavya Doshi

ART和Dalvik

How Android Apps are Built and Run

Build and Run

《程序员的自我修养—链接、装载与库》

《深入理解Java虚拟机》

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容