1.博客概要
本文詳細介紹了當今流行的一鍵生成APP技術。介紹了這種設計思想的來源,介紹了國內外的研究背景,並介紹了這個技術體系中的一些實現細節,歡迎各路大神們多提意見。一鍵生成技術,說的通俗點就是,要在自有平台上發布若干個代碼一致,但包名,版本,引用資源都不同的App,即實現一套代碼生成多個不同的APK(注意哦,這一系列的編譯打包都是自動化的)。
2.認識一個Android項目
Android是一種基於Linux的自由及開放源代碼的操作系統,主要使用於移動設備,如智能手機和平板電腦。
如上圖所示,是一個Android項目的目錄結構圖,分別介紹如下:
- src文件,即source code,項目源代碼。
- gen文件,該文件夾下面有個R.java文件,R.java是在建立項目時自動生成的,這個文件是只讀模式的,不能更改。R.java文件中定義了一個類R,R類中包含很多靜態類,且靜態類的名字都與res中的一個名字對應,即R類定義該項目所有資源的索引。通過R.java我們可以很快地查找我們需要的資源,另外編繹器也會檢查R.java列表中的資源是否被使用到,沒有被使用到的資源不會編繹進軟件中,這樣可以減少應用在手機占用的空間。它定義的每個資源值都是唯一的,不會和系統沖突。這個文件由ADT插件自動更新,當你編輯過Res文件后保存,這個類就會自動更新。
- Android2.1文件,該文件夾下包含android.jar文件,這是一個Java歸檔文件,其中包含構建應用程序所需的所有的Android SDK 庫(如Views、Controls)和APIs。通過android.jar將自己的應用程序綁定到Android SDK和Android Emulator,這允許你使用所有Android的庫和包,且使你的應用程序在適當的環境中調試。
- asset文件,包含應用系統需要使用到的諸如mp3、視頻類的文件。
- res文件,資源目錄,包含你項目中的資源文件並將編譯進應用程序。向此目錄添加資源時,會被R.java自動記錄。新建一個項目,res目錄下會有三個子目錄:drawabel、layout、values。drawabel:包含一些你的應用程序可以用的圖標文件(*.png、*.jpg) ,存在多個是為了適應不同分辨率。layout:界面布局文件,自定義的UI。values:軟件上所需要顯示的各種文字,可以存放多個*.xml文件,還可以存放不同類型的數據:比如arrays.xml、colors.xml、strings.xml、styles.xml。
- AndroidMainfest.xml文件,項目的總配置文件。項目的總配置文件,記錄應用中所使用的各種組件。這個文件列出了應用程序所提供的功能,在這個文件中,你可以指定應用程序使用到的服務(如電話服務、互聯網服務、短信服務、GPS服務等等)。另外當你新添加一個Activity的時候,也需要在這個文件中進行相應配置,只有配置好后,才能調用此Activity。
- default.properties文件,記錄項目中所需要的環境信息,比如Android的版本等。
3.Android項目是如何實現編譯、打包、簽名和發布的,這套流程是怎樣的?
知道了一個Android項目的組成之后,我們來看一個apk是如何從Eclipse中的源文件組裝成一個apk文件的。
如上圖所示,apk的編譯過程分為以下幾個階段:
1.用aapt命令生成R.java文件,命令“ aapt package -f -m -J ./gen -S res -M AndroidManifest.xml -I D:\android.jar ”
2.用aidl命令生成相應java文件。
3.用javac命令編譯java源文件生成class文件
4.用dx.bat將class文件轉換成classes.dex文件
5.用aapt命令生成資源包文件resources
6.用apkbuilder.bat打包資源和classes.dex文件,生成unsigned.apk
7.用jarsinger命令對apk認證,生成signed.apk
4.結合配置文件,詳細解釋一鍵生成APP是如何通過build.xml來自動化實現的?
從上面步驟,我們知道了通過一堆命令,可以依次來執行這七個步驟,是通過手動命令來編譯打包的。一鍵生成APP技術則是利用了Ant的自動編譯打包apk的技術,就是這么簡單。相對應的有如下步驟:
1.生成R.java文件
<!-- Generate the R.java file for this project's resources. --> <target name="resource-src" depends="copy"> <echo>Generating R.java / Manifest.java from the resources</echo> <exec executable="${aapt}" failonerror="true"> <arg value="package" /> <arg value="-m" /> <arg value="-J" /> <arg value="${outdir-gen}" /> <arg value="-M" /> <arg value="AndroidManifest.xml" /> <arg value="-S" /> <arg value="${resource-dir}" /> <arg value="-I" /> <arg value="${android-jar}" /> </exec> </target>
效果等價於,執行了aapt命令生成R.java文件。
2.將aidl文件生成Java文件
<!-- Generate java classes from .aidl files. --> <target name="aidl" depends="copy"> <echo>Compiling aidl files into Java classes...</echo> <apply executable="${aidl}" failonerror="true"> <arg value="-p${android-framework}" /> <arg value="-I${srcdir}" /> <fileset dir="${srcdir}"> <include name="**/*.aidl"/> </fileset> </apply> </target>
3.將.java類文件生成class文件
<!-- Compile this project's .java files into .class files. --> <target name="compile" depends="copy, resource-src, aidl"> <javac encoding="GB18030" target="1.5" debug="true" extdirs="" srcdir="." destdir="${outdir-classes}" bootclasspath="${android-jar}"> <classpath> <fileset dir="${external-libs}" includes="*.jar"/> </classpath> </javac> </target>
下面的步驟,依次類推,詳情參考:http://jojol-zhou.iteye.com/blog/729271
5.總結
<?xml version="1.0" ?> <!-- byread Package Utility Author: Jojol Zhou Date: 20100804 --> <project name="Byread" default="debug"> <!-- SDK Locations --> <property name="sdk2.2-folder" value="F:\explorer\android-sdk-windows2.2" /> <property name="sdk-folder" value="${sdk2.2-folder}/platforms/android-3" /> <property name="sdk-tools" value="${sdk-folder}/tools" /> <property name="android-tools" value="${sdk2.2-folder}/tools" /> <property name="proguardpath" location="${wtkhome}/lib/proguard" /> <!-- step 1.generate R.java by aapt 2.compile java to class by javac。exe 3.generate classes.dex by dx.bat --> <!-- 4.package resources by aapt 5. package resource and classes.dex by apkbuilder.bat 6.sign apk by jarsinger --> <!-- Tools --> <property name="aapt" value="${sdk-tools}/aapt.exe" /> <property name="dx" value="${sdk-tools}/dx.bat" /> <property name="apk-builder" value="${android-tools}/apkbuilder.bat" /> <property name="aidl" value="${android-tools}/aidl.exe" /> <property name="adb" value="${android-tools}/adb.exe" /> <property name="android-jar" value="${sdk-folder}/android.jar" /> <property name="jarsigner" value="C:\Program Files\Java\jdk1.6.0_07\bin\jarsigner.exe" /> <!-- Application Package Name --> <property name="application-package" value="com.byread.reader" /> <property name="useragent" value="byAndroidWeb" /> <property name="version" value="1.02" /> <!-- The intermediates directory --> <!-- Eclipse uses "bin" for its own output, so we do the same. --> <property name="outdir-bin" value="bin" /> <property name="outdir-gen" value="gen" /> <!-- source directories --> <property name="resource-dir" value="res" /> <property name="asset-dir" value="assets" /> <property name="srcdir" value="src" /> <property name="srcdir-ospath" value="${basedir}/${srcdir}" /> <property name="external-libs" value="libs" /> <property name="external-libs-ospath" value="${basedir}/${external-libs}" /> <!-- dest directories --> <property name="des-resource-dir" value="${outdir-bin}/res" /> <property name="des-asset-dir" value="${outdir-bin}/assets" /> <property name="des-srcdir" value="${outdir-bin}/src" /> <property name="des-srcdir-ospath" value="${basedir}/${outdir-bin}/${srcdir}" /> <property name="des-external-libs" value="${outdir-bin}/libs" /> <property name="des-external-libs-ospath" value="${basedir}/${outdir-bin}/${external-libs}" /> <!-- Output directories --> <property name="outdir-classes" value="${outdir-bin}/src" /> <property name="outdir-obfuscate-classes" value="${outdir-bin}/classes" /> <property name="outdir-obfuscate-classes-ospath" value="${basedir}/${outdir-obfuscate-classes}" /> <!-- Intermediate files --> <property name="dex-file" value="classes.dex" /> <property name="intermediate-dex" value="${outdir-bin}/${dex-file}" /> <property name="intermediate-dex-ospath" value="${basedir}/${intermediate-dex}" /> <!-- The final package file to generate --> <property name="resources-package" value="${outdir-bin}/${ant.project.name}" /> <property name="resources-package-ospath" value="${basedir}/${resources-package}" /> <property name="out-debug-package" value="${outdir-bin}/${ant.project.name}-debug.apk" /> <property name="out-debug-package-ospath" value="${basedir}/${out-debug-package}" /> <property name="out-unsigned-package" value="${outdir-bin}/${ant.project.name}-unsigned.apk" /> <property name="out-unsigned-package-ospath" value="${basedir}/${out-unsigned-package}" /> <property name="out-signed-package" value="${useragent}\${ant.project.name}.apk" /> <property name="out-signed-package-ospath" value="${basedir}\${out-signed-package}" /> <!-- init --> <target name="init"> <echo>Creating all output directories </echo> <delete dir="${outdir-bin}" /> <delete dir="${useragent}" /> <mkdir dir="${outdir-bin}" /> <mkdir dir="${outdir-classes}" /> <mkdir dir="${useragent}" /> </target> <!-- copy original strings and modify useragent --> <target name="copy" depends="init"> <echo>copy files to output folder</echo> <delete file="${resource-dir}\values\strings.xml"/> <copy file="strings.xml" todir="${resource-dir}\values" /> <replace file="${resource-dir}\values\strings.xml" token="@USERAGENT@" value="${useragent}" encoding="utf-8"/> </target> <!-- Generate the R.java file for this project's resources. --> <target name="resource-src" depends="copy"> <echo>Generating R.java / Manifest.java from the resources...</echo> <exec executable="${aapt}" failonerror="true"> <arg value="package" /> <arg value="-m" /> <arg value="-J" /> <arg value="${outdir-gen}" /> <arg value="-M" /> <arg value="AndroidManifest.xml" /> <arg value="-S" /> <arg value="${resource-dir}" /> <arg value="-I" /> <arg value="${android-jar}" /> </exec> </target> <!-- Generate java classes from .aidl files. --> <target name="aidl" depends="copy"> <echo>Compiling aidl files into Java classes...</echo> <apply executable="${aidl}" failonerror="true"> <arg value="-p${android-framework}" /> <arg value="-I${srcdir}" /> <fileset dir="${srcdir}"> <include name="**/*.aidl"/> </fileset> </apply> </target> <!-- Compile this project's .java files into .class files. --> <target name="compile" depends="copy, resource-src, aidl"> <javac encoding="GB18030" target="1.5" debug="true" extdirs="" srcdir="." destdir="${outdir-classes}" bootclasspath="${android-jar}"> <classpath> <fileset dir="${external-libs}" includes="*.jar"/> </classpath> </javac> </target> <!-- Convert this project's .class files into .dex files. --> <target name="dex" depends="compile"> <echo>Converting compiled files and external libraries into ${outdir}/${dex-file}...</echo> <apply executable="${dx}" failonerror="true" parallel="true"> <arg value="--dex" /> <arg value="--output=${intermediate-dex-ospath}" /> <arg path="${outdir-obfuscate-classes-ospath}" /> <fileset dir="${external-libs}" includes="*.jar"/> </apply> </target> <!-- Put the project's resources into the output package file. --> <target name="package-res-and-assets"> <echo>Packaging resources and assets...</echo> <exec executable="${aapt}" failonerror="true"> <arg value="package" /> <arg value="-f" /> <arg value="-M" /> <arg value="AndroidManifest.xml" /> <arg value="-S" /> <arg value="${resource-dir}" /> <arg value="-A" /> <arg value="${asset-dir}" /> <arg value="-I" /> <arg value="${android-jar}" /> <arg value="-F" /> <arg value="${resources-package}" /> </exec> </target> <!-- Same as package-res-and-assets, but without "-A ${asset-dir}" --> <target name="package-res-no-assets"> <echo>Packaging resources...</echo> <exec executable="${aapt}" failonerror="true"> <arg value="package" /> <arg value="-f" /> <arg value="-M" /> <arg value="AndroidManifest.xml" /> <arg value="-S" /> <arg value="${resource-dir}" /> <!-- No assets directory --> <arg value="-I" /> <arg value="${android-jar}" /> <arg value="-F" /> <arg value="${resources-package}" /> </exec> </target> <!-- Invoke the proper target depending on whether or not an assets directory is present. --> <!-- TODO: find a nicer way to include the "-A ${asset-dir}" argument only when the assets dir exists. --> <target name="package-res"> <available file="${asset-dir}" type="dir" property="res-target" value="and-assets" /> <property name="res-target" value="no-assets" /> <antcall target="package-res-${res-target}" /> </target> <!-- Package the application and sign it with a debug key. This is the default target when building. It is used for debug. --> <target name="debug" depends="dex, package-res"> <echo>Packaging ${out-debug-package}, and signing it with a debug key...</echo> <exec executable="${apk-builder}" failonerror="true"> <arg value="${out-debug-package-ospath}" /> <arg value="-z" /> <arg value="${resources-package-ospath}" /> <arg value="-f" /> <arg value="${intermediate-dex-ospath}" /> <arg value="-rf" /> <arg value="${srcdir-ospath}" /> <arg value="-rj" /> <arg value="${external-libs-ospath}" /> </exec> </target> <!-- Package the application without signing it. This allows for the application to be signed later with an official publishing key. --> <target name="package" depends="dex, package-res"> <echo>Packaging ${out-unsigned-package} for release...</echo> <exec executable="${apk-builder}" failonerror="true"> <arg value="${out-unsigned-package-ospath}" /> <arg value="-u" /> <arg value="-z" /> <arg value="${resources-package-ospath}" /> <arg value="-f" /> <arg value="${intermediate-dex-ospath}" /> <arg value="-rf" /> <arg value="${srcdir-ospath}" /> <arg value="-rj" /> <arg value="${external-libs-ospath}" /> </exec> <echo>It will need to be signed with jarsigner before being published.</echo> </target> <target name="jarsigner" depends="package"> <echo>Packaging ${out-unsigned-package} for release...</echo> <exec executable="${jarsigner}" failonerror="true"> <arg value="-verbose" /> <arg value="-storepass" /> <arg value="byread002" /> <arg value="-keypass" /> <arg value="byread002" /> <arg value="-keystore" /> <arg value="bbyread.keystore" /> <arg value="-signedjar" /> <arg value="${out-signed-package-ospath}" /> <arg value="${out-unsigned-package-ospath}" /> <arg value="byread" /> </exec> </target> <target name="release" depends="jarsigner"> <echo>release for release...</echo> </target> <!-- Install the package on the default emulator --> <target name="install" depends="debug"> <echo>Installing ${out-debug-package} onto default emulator...</echo> <exec executable="${adb}" failonerror="true"> <arg value="install" /> <arg value="${out-debug-package}" /> </exec> </target> <target name="reinstall" depends="debug"> <echo>Installing ${out-debug-package} onto default emulator...</echo> <exec executable="${adb}" failonerror="true"> <arg value="install" /> <arg value="-r" /> <arg value="${out-debug-package}" /> </exec> </target> <!-- Uinstall the package from the default emulator --> <target name="uninstall"> <echo>Uninstalling ${application-package} from the default emulator...</echo> <exec executable="${adb}" failonerror="true"> <arg value="uninstall" /> <arg value="${application-package}" /> </exec> </target> </project>
參考文獻:
[1] http://www.cnblogs.com/qianxudetianxia/archive/2012/07/04/2573687.html
[2]