android的编译和运行过程深入分析

   2015-09-05 0
核心提示:首先来看一下使用Java语言编写的Android应用程序从源码到安装包的整个过程,此过程对了解android的编译和运行过程有很大的帮助

首先来看一下使用Java语言编写的Android应用程序从源码到安装包的整个过程,示意图如下,其中包含编译、链接和签名等:

android的编译和运行过程深入分析

(1)使用aapt工具生成R.java文件

可以先通过搭建好的Eclipse开发环境创建一个未编译的Android工程,记的一定要将Eclipse中Project菜单下的Build Automatically选项前面的对勾去掉后再去创建工程。创建好未编译的工程后,在命令行中输入如下命令:

d:\android-sdk-windows\platform-tools>aapt package -f -m -M "C:\Documents and Settings\******\workspace\HelloAndroid3\AndroidManifest.xml" -J "C:\Documents and Settings\******\workspace\HelloAndroid3\gen" -S "C:\Documents and Settings\******\workspace\HelloAndroid3\res" -I "D:\android-sdk-windows\platforms\android-10\android.jar"

其中-M及紧跟其后的参数是用于指定AndroidManifest.xml(配置文件)的路径,-J及紧跟其后的参数是指定R.java生成路径,-S及后面参数是指定资源文件所在目录,-I及后面参数是指定要包含的Android平台类库;运行后会在工程目录中的gen目录下生成R.java文件。aapt的具体用法可在命令行输入aapt后会看到。

R.java文件的作用是提供给程序访问资源的入口,更详细的内容请参见后面关于Android工程的文件结构和详解的博文。

(2)使用aidl工具将.aidl文件编译成.java文件

AIDL是Android系统提供的一种进程间调用的方式,类似于IPC调用,通过aidl工具将使用Android Interface Definition Language描述的.aidl文件编译成包含java接口类的.java文件,然后进程间遵循这些接口进行相互调用。.aidl文件一般与程序源码文件存放在一起。对于该例子中自动创建的工程来说,没有用到AIDL,所以不进行这一步。aidl工具的用法如下:

usage: aidl OPTIONS INPUT [OUTPUT]
       aidl --preprocess OUTPUT INPUT...

OPTIONS:
   -I<DIR>    search path for import statements.
   -d<FILE>   generate dependency file.
   -p<FILE>   file created by --preprocess to import.
   -o<FOLDER> base output folder for generated files.
   -b         fail when trying to compile a parcelable.

INPUT:
   An aidl interface file.

OUTPUT:
   The generated interface files.
   If omitted and the -o option is not used, the input filename is used, with the .aidl extension changed to a .java extension.
   If the -o option is used, the generated files will be placed in the base output folder, under their package folder

(3)使用javac工具将.java文件编译成.class文件

d:\Java\jdk1.6.0_25\bin>javac -encoding GB18030 -target 1.6 -bootclasspath "D:\android-sdk-windows\platforms\android-10\android.jar" -d "C:\Documents and Settings\******\workspace\HelloAndroid3\bin" "C:\Documents and Settings\******\workspace\HelloAndroid3\src\com\******\HelloAndroid3\HelloAndroid3.java" "C:\Documents and Settings\******\workspace\HelloAndroid3\gen\com\******\HelloAndroid3\R.java"

期间,我本来想使用*.java来描述需要编译的源码文件,但提示找不到,后来将源码文件指定为具体的HelloAndroid3.java文件后才编译通过,奇怪。

随后会在工程目录下的bin目录下生成.class文件。

(4)使用dx.bat批处理将众多.class文件转换成一个.dex文件

D:\android-sdk-windows\platform-tools>dx --dex --output=c:\docume~1\******\workspace\HelloAndroid3\bin\classes.dexc:\docume~1\******\workspace\HelloAndroid3\bin\

--output及后面的路径指明.dex文件的生成路径;红色标注的路径为.class所在的路径,需要注意的是,这里不能加上包路径,否则会报不匹配的错误,可能在批处理中已将添加包路径。另外,如遇windows系统路径含有空格的话一律使用缩写形式,具体有哪些系统路径及其缩写是什么,还是问度娘吧。成功后便在指定路径下生成了.dex文件。.dex文件是在Android的Dalvik虚拟机上运行的,具体内容后面的运行原理会提到。

(5)使用aapt工具打包资源文件

D:\android-sdk-windows\platform-tools>aapt package -f -M C:\Docume~1\******\workspace\HelloAndroid3\AndroidManifest.xml -S C:\Docume~1\******\workspace\HelloAndroid3\res -A C:\Docume~1\******\workspace\HelloAndroid3\assets -I D:\android-sdk-windows\platforms\android-10\android.jar -F C:\Docume~1\******\workspace\HelloAndroid3\bin\resources.ap_

对照R.java文件的生成,可以看到参数发生了变化,少了-m 和 -J,如果看aapt用法中的描述就知道,-m和-J是结对出现的,用以指明R.java文件的生成路径。-M、-S、-I之前都有提到,这里不再介绍。-F的作用是指明打包后的资源文件的路径,在最后一定要加上文件名,最好加上扩展名。这里参考Eclipse中自动编译时制定的.ap_后缀名。

(6)使用apkbuilder生成未签名的apk安装文件

D:\android-sdk-windows\tools>apkbuilder C:\Docume~1\******\workspace\HelloAndroid3\bin\HelloAndroid3.apk -v -u -z C:\Docume~1\******\workspace\HelloAndroid3\bin\resources.ap_ -f C:\Docume~1\******\workspace\HelloAndroid3\bin\classes.dex -rf C:\Docume~1\******\workspace\HelloAndroid3\src

其中,apkbuilder后面紧跟的路径是生成的apk安装文件的路径,-v参数的作用是指明执行中输出必要信息,具体输出内容如下:

Packaging HelloAndroid3.apk
C:\Docume~1\******\workspace\HelloAndroid3\bin\resources.ap_:
=> res/layout/main.xml
=> AndroidManifest.xml
=> resources.arsc
=> res/drawable-hdpi/icon.png
=> res/drawable-ldpi/icon.png
=> res/drawable-mdpi/icon.png
C:\Docume~1\******\workspace\HelloAndroid3\bin\classes.dex => classes.dex

-u参数表示生成的是未签名的安装包,-z及后面的路径表明打包了的资源文件的路径,-f及后面的路径指明了.dex文件的路径,-rf指明了源文件的目录。

(7)使用jdk中的jarsigner对apk安装文件进行签名

签名的目的是保证应用程序的开发者的唯一性,签名需要的东西除了jarsigner工具外还有密钥文件,即.keystore文件,我们这里不产生自己的keystore文件,而是采用Android SDK提供的Debug.keystore文件,其位置是在“我的文档”下的.android目录下。签名的原理及密钥文件的产生等内容在后续的博文中补充。

D:\Java\jdk1.6.0_25\bin>jarsigner -keystore C:\Docume~1\******\.android\debug.keystore -storepass android -keypass android -signedjar C:\Docume~1\******\workspace\HelloAndroid3\bin\Hello3.apk C:\Docume~1\******\workspace\HelloAndroid3\bin\HelloAndroid3.apk androiddebugkey

-keystore及后面的路径指明密钥文件的位置,-storepass是用于密钥库完整性的口令,-keypass是专用密钥的口令,-signedjar及后面的路径指明签完名的apk文件的路径,紧接着的是需要签名的apk的路径,最后面是密钥的别名。debug.keystore的name和passwords信息是在SDK文档中找到的,具体内容如下:

The SDK tools create the debug keystore/key with predetermined names/passwords:

Keystore name: "debug.keystore" Keystore password: "android" Key alias: "androiddebugkey" Key password: "android" CN: "CN=Android Debug,O=Android,C=US"
 
标签: 编译 运行 过程
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • Intellij IDEA 中无法编译lombok的解决方法
    Intellij IDEA 中无法编译lombok的解决方法
    新装了Intellij IDEA,导入项目之后,引入各种库之后,却一直报错,锁定错误,发现提示Bean中的属性没有getter和setter方法。这个是使用lombok的,lombok的jar包也已经引入。可是还是报错,后来查了一些原因,原来还需要安装plugins。步骤如下:1、右上角打开
  • 【ART世界探险】Android N上的编译流程解析
    【ART世界探险】Android N上的编译流程解析
    ART世界探险(20) - Android N上的编译流程就在我们分析Android M版本的ART还只走出了一小段路的时候,Android N的新ART就问世了。Android N上的ART还是有不小的改进的。不过做为一个关注细节的系列文章,我们还是从Compile的过程说起。流程概述在安装的时候,
    01-06 安卓开发
  • Android Apk 文件反编译和重新打包的过程分析
    Android Apk 文件反编译和重新打包的过程分析
    学习的方式有很多种,Read the fucking source code 绝对不失为最佳的方式。除了阅读 Android SDK 的源码,还能通过反编译 APK 的方式,阅读他人开发的应用源码。本文就来聊聊在 Mac 环境下如何借助第三方工具反编译 APK 压缩文件(通过爱加密、360加固等手段
  • Android编译时注解实践指南
    Android编译时注解实践指南
    Android注解分为两种,一种是运行时注解,一种是编译时注解。RxJava就是运行时注解,而butterKnife和EventBus是编译时注解,啃代码的时候经常碰到注解,所以只能乖乖的学习。本文主要讲解演示如何在Android Studio上运行一个编译时注解的Demo,被注解的对象打
  • 每日一博|深入理解 gradle 编译- Android 基础篇
    每日一博|深入理解 gradle 编译- Android 基础
    9/27/2016 1:28:27 PM深入理解gradle编译-Android基础篇导读Gradle基于Groovy的特定领域语言(DSL)编写的一种自动化建构工具,Groovy作为一种高级语言由Java代码实现,本文将对Gradle一些常见问题进行一一介绍: 理解Gradle与android app之间的关系,以及Gra
  • Android系统篇之----编写简单的驱动程序并且将其编译到内核源码中
    Android系统篇之----编写简单的驱动程序并且将
    通过之前的一篇文章,我们了解了 Android中的Binder机制和远程服务调用 在这篇文章中主要介绍了Android中的应用在调用一些系统服务的时候的原理,那么接下来就继续来介绍一下如何通过编译Android源码来手动添加一个系统服务,让编译之后的Android系统中存在我
  • Android-NDK:设置NDK编译选项
    Android-NDK:设置NDK编译选项
    在Android NDK 开发中,编译选项配置能够帮助我们很好的使用NDK/C/C++的特性,这些编译选项中那个包括:APP_ABI,LOCAL_LDLIBS, LOCAL_CFLAGS, APP_STL ...等,下面是这些选项中的一些配置.选项概述(一) Android.mk选项 解释 LOCAL_MODULE 模块名 LOCAL_SRC_FILES
  • 手把手图文并茂教你用 Android Studio 编译 FFmpeg 库并移植
    手把手图文并茂教你用 Android Studio 编译 FFm
    之前曾写过一篇 《FFmpeg在Linux下安装编译过程》 ,今天有空就用Android Studio编译FFmpeg库并移植,并附上调用ffmpeg简单示例,看下Agenda如下: 下载FFmpeg 3.1.3 下载NDK 配置ndk环境 编译FFmpeg 移植到Android平台 调用编译的so示例 下载FFmpeg 3.1.3网址
  • [原]编译开源LibreOffice的Android版本
    项目近期需要用到文档查看器,现有的WPS之类确实比较好用,无奈还需要对应用做些处理,只好寻找开源的实现。开源的文档查看器主要两种方式,一种是完整的Android版本实现,直接拿来即用,如LibreOffice的Android版本;一种是提供了Java库之类,能够解析标准的
  • Android反编译技术总结
    Android反编译技术总结
    一、Apk反编译工具及其使用方法1.原理学习反编译之前,建议先学习一下Apk打包的过程,明白打包完成后的Apk里面都有什么文件,各种文件都是怎么生成的。这里有两篇AndroidWeekly中推荐过的好文章: 浅析 Android 打包流程 Android构建过程分析Apk技术也有非常
点击排行