YangMing +

android编译原理

所有代码研究基于android4.2.2_R1

最近新的项目由我来些android的打包脚本所以特此研究了下android打包的过程。

android build process

Alt text

基于的tools都再{androidsdk}/tool/ 和 {androidsdk}/platforms/android目录下面 打包过程简析(我只是官网的搬运工) - 使用Android Asset Packaging Tool(aapt) ,将AndroidManifest.xml和res下的资源编译生成R.java文件,这样java文件就可以去引用资源了 - 使用aidl 工具去生成对应的Java interfaces - 将src和通过aapt生成的R.java,.aidl文件通过javaC命令去生成.class 文件 - 使用dex tool 将class文件转化成Dalvik byte code.这时候要将所有class文件和第三方的jar包都包括。 - 所有没有编译过得图片和编译过的图片,.dex文件传给apkbuilder去打包成.apk - 最后采用zipalign tool 打入签名

这个就是一次打包要走的完整流程,打包脚本目前分ant和gradle一个是ant配合eclipse另一个是android studio 配合gradle。先说说ant打包

ant脚本

生成项目的build.xml

Eclipse中使用Ant为Android打包并且签名。 SDK自带文件 在tools/ant目录下这3个文件。 - build.xml - NOTICE - uibuild.xml 这里头定义了大量基础构建方法和打包策略,我们只需要建立自己的build.xml并存放在项目的根目录,然后引用下这个build.xml。

我们可以使用sdk自带的android.bat去为项目生成对应的build.xml

通过执行如下命令即可在指定的项目目录下生成build.xml文件。 /tools/android update project -p -t

其中: sdk为sdk的安装目录,其下的tools/android是我们要使用的命令。 project为项目目录。 target为项目所使用的android的target id,也就是项目对应的android的版本。 可以通过执行以下命令查看当前sdk中所包含的target以及相应id: android list targets。

使用ant打android 的jar包

<?xml version="1.0" encoding="UTF-8"?>
<!--project 用于定义一个ant工程,其中的三项name、default、basedir缺一不可。
作用分别为:定义工程名、制定默认执行的任务、以及工程基础的路径型(它是计算其它路径的基础,一般情况下使用.即在java工程根目录即可)-->
<project name="sayhellousejarant" default="compile" basedir=".">
    <!--描述,个人觉得就是一提示作用,没什么实际用途-->
    <description>use jar test</description>
    <!--定义源文件路径,其中的value换成location也行,使用value的时候,${src}得到的就是src这个值,如果使用location,得到的是src这个目录的绝对路径-->
    <property name="src" value="src" />
    <property name="classes" value="bin/classes" />

    <!--构造打包时Class-Path需要的路径 -->
    <!--pathconvert用于对目录进行组合 property即这个组合的名字,pathsep作用是各个文件之间的分隔符,
        如果不写,在windows平台默认是分号。但时在MANIFEST.MF这个文件中,各个jar包之间要用空格区分,
        因此。这里就写成空格了
    -->
    <pathconvert property="lib" pathsep=" ">
        <!--mapper,对路径组合方式进行控制-->
        <mapper>
            <!--chainedmapper 作用是联合多个mapper-->
            <chainedmapper>
                <!--过滤文件,将路径去掉,只保留文件名-->
                <flattenmapper />
                <!--过滤+转换器,将所有的文件名前面都加上一个lib,我们知道lib目录下面有jar包,
                    lib/*的作用其实是将jar包名与路径进行组合形成如:lib/google.jar这样的相对路径
                 -->
                <globmapper from="*" to="lib/*" />
            </chainedmapper>
        </mapper>
        <!--按照mapper定义的格式组合lib目录下面的所有jar文件,形成诸如lib/jar1.jar lib/jar2.jar的字符串-->
        <fileset dir="lib">
            <include name="*.jar" />
        </fileset>
    </pathconvert>


    <!--同lib,此处不再解释-->
    <pathconvert property="lib2" pathsep=" ">
        <mapper>
            <chainedmapper>
                <flattenmapper />
                <globmapper from="*" to="lib2/*" />
            </chainedmapper>
        </mapper>
        <fileset dir="lib2">
            <include name="*.jar" />
        </fileset>
    </pathconvert>

    <!--单独一个jar包,不在lib以及lib2目录下,使用一个单独的property定义,以便引用-->
    <property name="androidjar" value="android-201111262247.jar" />
    <!--组合各个路径,构成MANIFEST.MF文件中Class-Path所需的字符串-->
    <property name="libs" value="${lib} ${lib2} ${androidjar}" />

    <!--打印一下刚才构造好的字符串,看看是否符合要求-->
    <echo>libs   ${libs}</echo>

    <!-- 构造打包时Class-Path需要的路径 结束-->

    <!--创建任务init,负责初始化一些条件-->
    <target name="init">
        <!-- 创建存放编译后的class的目录
            mkdir可以创建多级目录 
        -->
        <mkdir dir="${classes}" />
    </target>

    <!--创建编译任务,名字是compile,depends指定了comiple任务依赖init任务-->
    <target name="compile" depends="init" description="comile target">
        <!--javac,编译,对应java中的javac命令。
        其中srcdir定义源文件路径 destdir定义编译后文件路径,
        includeantruntime作用是指定编译任务是否包含ant的classpath,可有可无,不影响编译,
        但不写可能会出现警告,为了眼不见心不烦,加上吧-->
        <javac srcdir="${src}" destdir="${classes}" includeantruntime="true">
            <!-- classpath 定义编译需要的claspath -->
            <classpath>
                <fileset dir="lib">
                    <include name="*.jar" />
                </fileset>
                <fileset dir="lib2">
                    <include name="*.jar" />
                </fileset>
                <fileset dir=".">
                    <include name="${androidjar}" />
                </fileset>
            </classpath>
        </javac>
    </target>


    <!-- 创建时间戳 -->
    <tstamp />

    <!--定义jarfilename,准备进行打包操作。其中ant.project.name是ant默认的一个变量,值为最上面定义的project的name
    ${DSTAMP}为日期,格式为20111123;${TSTAMP}为时间,格式为2256,表示22点56分。
        -->
    <property name="jarfilename" value="${ant.project.name}-${DSTAMP}${TSTAMP}.jar" />
    <!--打包开始,名字为jar,依赖任务为compile-->
    <target name="jar" depends="compile" description="make jar file">
        <!--jar操作,jarfile指定jar包存放路径,basedir为编译后的class的目录-->
        <jar jarfile="${jarfilename}" basedir="${classes}">
            <!--为jar包指定manifest,当然,如果jar包不需要打成runnable的形式,manifest可以不要-->
            <manifest>
                <!--指定main-class-->
                <attribute name="Main-Class" value="demo.SayHello" />
                <!--指定Class-Path-->
                <attribute name="Class-Path" value="${libs}">
                </attribute>
            </manifest>
        </jar>
    </target>

    <!--运行一下jar包,试试看效果-->
    <target name="run" depends="jar">
        <!--其实这里就是运行jar命令,注意fork一定加上,不然不起作用-->
        <java jar="${jarfilename}" fork="true">
        </java>
    </target>


    <!-- 清理 -->
    <target name="clean">
        <!-- 可以以递归的方式删除目录 -->
        <delete dir="${classes}" />
        <delete dir="." includes="${ant.project.name}*.jar" />
    </target>
</project>

这个是网上的一个实例代码。其实打jar包主要会用导两个命令,一个是javac去生成class,一个是用jar命令去生成jar包.

gradle脚本 (待续)

评论:

Blog

Opinion

Project