ant build.xml 打包应三方jar注意的问题与混淆R的写法

老规矩我们还是来看看说在前面的话:首先我们得分清楚android在打包成apk的过程中要经过哪几个步骤:

Android编译的具体流程如下:

1)  ndk-build编译native代码生成so文件

2)  aapt命令根据res资源文件生成R.java

3)  aidl命令解析.aidl文件生成对应java文件

4)  javac命令编译java文件为class文件

5)  dx命令将编译好的class文件打包为dex文件

6)  aapt命令根据res资源文件生成resources.ap_资源索引文件

7)  apkbuilder将resources.ap_与dex打包成未签名apk文件

8)  jarsinger签名apk文件

9)  zipalign工具优化apk读取速度

这里我想重点说的是javac这个阶段和aapt打包成res文件混淆的问题。

这里我们看看假如我们的工程里面如果应用到了第三方的jar应该怎么写,我们首先需要把我们的jar包复制到我们的libs下面然后修改一下编译这里的脚本,假如这个外部工程还有自己的R文件和res资源我们在写脚本的时候需要注意这几个问题.

1:编译成R文件的时候记得一起打包这个外部依赖项目的R文件。

2:编译class文件的时候记得加入lib依赖

3:打包res文件的时候记得外部文件的res也要添加

4:如果项目中有混淆记得不要混淆外部依赖文件的R文件,不然找不到外部的R

这里我来举个例子吧,我这里例如我用到了,PullrefreshListview这个外部依赖工程,需要注意的是这几个步骤,我贴出脚本吧。先看看编译R文件的时候的脚本。

<!-- 为该项目资源生成R.java文件 -->
	<target name="gen" depends="init">
		<echo>从资源文件生成R.java ...</echo>
		<exec executable="${aapt}" failonerror="true">
			<arg value="package" />
			<span style="color:#ff0000;"><arg value="--auto-add-overlay"></arg></span>
			<arg value="-m" />
			<arg value="-J" />
			<arg value="${gen}" />
			<arg value="-M" />
			<arg value="AndroidManifest.xml" />
			<arg value="-S" />
			<arg value="res" />
			<span style="color:#ff0000;"><arg value="-S" />
			<arg value="C:\Users\edsheng\Desktop\Android-PullToRefresh-master\library\res" />
			<arg value="--extra-packages" />
			<arg value="com.handmark.pulltorefresh.library"/></span>
			<arg value="-I" />
			<arg value="${androidjar}" />
		</exec>
		<echo>R.java文件生成成功</echo>
	</target>

看到我红色的部分了吗?这些就是外部依赖工程的时候需要添加的脚本是把我们外部依赖的res文件拿去生成R,文件。

然后需要注意的第二部分,编译class文件记得加入jar的依赖,我们要把用到的jar先复制到lib里面,由于Pullrefreshlistview这个工程是个lib工程,我们直接拷贝lib到我们的主工程的libs文件目录下边然后看看javac编译class文件的脚本。

<target name="compile" depends="aidl">
		<echo>开始编译.class文件...</echo>
		<javac fork="true" executable="${javac}" encoding="${encoding}" debug="true" extdirs="" source="1.5" target="1.5" destdir="${classes}"  bootclasspath="${androidjar}">
            <src path="${src}" />
            <src path="${gen}" />
      <span style="color:#ff0000;">      <classpath>
                <fileset dir="${root}/libs">
                    <include name="*.jar" />
                </fileset>
                </classpath></span>
        </javac>
		<echo>.class文件编译完成</echo>
	</target>

也是要理解这个编译过程要找libs下面的所有jar,同理然后我们来看看混淆代码的时候要注意的东西,一个就是lib文件一个就是混淆的时候lib的资源R文件不要去混淆了,这里我也贴出自己的混淆的脚本。

<target name="obfuscate" depends="compile">

        <echo>对打包的结果进行混淆...</echo>
        <java jar="${proguard_home}/lib/proguard.jar" fork="true" failonerror="true">

            <jvmarg value="-Dmaximum.inlined.code.length=32" />

            <arg value="-injars ${classes}" />

            <arg value="-outjars obfuscated.jar" />

            <arg value="-libraryjars ${androidjar}" />

 	<span style="color:#ff0000;">    <arg value="-libraryjars C:\Users\edsheng\Desktop\wecheckscore\libs\library.jar" />
            <!-- 如果使用到外部库,请在这里指定 -->

			<arg value="@proguard.cfg" />
 			<arg value="-dump ${out}/dump.txt"/>
			<arg value="-printusage ${out}/usage.txt"/>
			<arg value="-printmapping ${out}/mapping.txt"/>
			<arg value="-printseeds ${out}/seeds.txt"/></span>

	 </java>
           <delete dir="${classes}"/>
           <mkdir dir="${classes}"/>
           <unzip src="obfuscated.jar" dest="${classes}"/>
            <delete file="obfuscated.jar"/>
	 <echo>代码混淆结束</echo>
	 </target>

需要注意的地方也是我红色的地方,指定外部混淆的jar,还有的一个是proguard.cfg这个文件,就是我们配置的文件我们还是打开这个文件看看,该怎么配置呢?

#-optimizationpasses 7
#-optimizations !code/simplification/arithmetic,!field/*,!class/merging/*
-dontoptimize
-verbose
-dontskipnonpubliclibraryclasses
-dontskipnonpubliclibraryclassmembers
#-overloadaggressively

<span style="color:#ff0000;">-keep public class  com.handmark.pulltorefresh.library.R*{*;}</span>

#------------------  下方是android平台自带的排除项,这里不要动         ----------------
-keep public class * extends android.app.Activity{
	public <fields>;
	public <methods>;
}
-keep public class * extends android.app.Application
{
	public <fields>;
	public <methods>;
}

-keep public class * extends android.app.Service
-keep public class * extends android.content.BroadcastReceiver
-keep public class * extends android.content.ContentProvider
-keep public class * extends android.app.backup.BackupAgentHelper
-keep public class * extends android.preference.Preference

-keepclassmembers enum * {
    public static **[] values();
    public static ** valueOf(java.lang.String);
}

-keepclasseswithmembers class * {
	public <init>(android.content.Context, android.util.AttributeSet);
}

-keepclasseswithmembers class * {
	public <init>(android.content.Context, android.util.AttributeSet, int);
}

-keepattributes *Annotation*

# 改成这样,因为发现部分SDK中没有被java代码显示调用过,SO文件中调用的native方法会被混淆掉

-keepclasseswithmembers class * {
	native <methods>;
}

-keep class * implements android.os.Parcelable {
  public static final android.os.Parcelable$Creator *;
}

#------------------  下方是共性的排除项目         ----------------
# 方法名中含有“JNI”字符的,认定是Java Native Interface方法,自动排除
# 方法名中含有“JRI”字符的,认定是Java Reflection Interface方法,自动排除

-keepclasseswithmembers class * {
    ... *JNI*(...);
}

-keepclasseswithmembernames class * {
	... *JRI*(...);
}

-keep class **JNI* {*;}

需要注意的也是红色的部分,提示了不要混淆lib的R文件,最后就是我们的res文件资源的打包了,然后我们也来看看我们的脚本应该是怎么写的吧。

!-- 打包项目的资源文件 -->
	<target name="package_res_with_assets">
		<echo>打包资源和资产文件...</echo>
		<exec executable="${aapt}" failonerror="true">
			<arg value="package" />
			<span style="color:#ff0000;">  <arg value="--auto-add-overlay"></arg></span>
			<arg value="-f" />
			<arg value="-M" />
			<arg value="AndroidManifest.xml" />
			<arg value="-S" />
			<arg value="res" />
			<span style="color:#ff0000;"><arg value="-S" />
			<arg value="C:\Users\edsheng\Desktop\Android-PullToRefresh-master\library\res" />
			 <arg value="--extra-packages" />
			 <arg value="com.handmark.pulltorefresh.library"/></span>
			<arg value="-A" />
			<arg value="assets" />
			<arg value="-I" />
			<arg value="${androidjar}" />
			<arg value="-F" />
			<arg value="${out}/${file_name}.ap_" />
		</exec>
		<echo>打包资源和资产文件完成</echo>
	</target>

还是红色的地方也是指定了外部的资源和lib这样外部工程的资源也被打包进来了。这样就可以不管是外部依赖一个还是多个都可以搞定了。那么大家就可以愉快的buld了。

时间: 2025-01-16 09:00:22

ant build.xml 打包应三方jar注意的问题与混淆R的写法的相关文章

利用ant build.xml打包一个project

项目结构如下: build.xml <project default="jar" name="test"> <property name="src" value="src" /> <property name="class" value="bin" /> <property name="output" value="

ant打包出错 ant\build.xml:698: null returned: 1

1.ant编译时突然报错:E:\android-sdk\tools\ant\build.xml:657:  The following error occurred while executing this line: null returned: 1 2.直接修改了sdk里面的配置,个人觉得不太妥,那么可以在自己项目的根目录下build.xml文件中加入: <property name="aapt.ignore.assets" value="!.svn:!.git:\

cocos2dx 编译错误 BUILD FAILEDG:\android\SDK\android-sdk-windows\tools\ant\build.xml:645: The following

BUILD FAILEDG:\android\SDK\android-sdk-windows\tools\ant\build.xml:645: The following erroroccurred while executing this line:G:\android\SDK\android-sdk-windows\tools\ant\build.xml:683: null returned: 1 解决办法: rd bin /S /Q cocos2dx 编译错误 BUILD FAILEDG:

Ant build.xml详解

Ant的概念 可能有些读者并不连接什么是Ant以及入可使用它,但只要使用通过Linux系统得读者,应该知道make这个命令.当编译Linux内核及一些软件的源程序时,经常要用这个命令.Make命令其实就是一个项目管理工具,而Ant所实现功能与此类似.像make,gnumake和nmake这些编译工具都有一定的缺陷,但是Ant却克服了这些工具的缺陷.最初Ant开发者在开发跨平台的应用时,用样也是基于这些缺陷对Ant做了更好的设计. Ant 与 makefile Makefile有一些不足之处,比如

Ant build.xml

Ant的概念可能有些读者并不连接什么是Ant以及入可使用它,但只要使用通过Linux系统得读者,应该知道make这个命令.当编译Linux内核及一些软件的源程序时,经常要用这个命令.Make命令其实就是一个项目管理工具,而Ant所实现功能与此类似.像make,gnumake和nmake这些编译工具都有一定的缺陷,但是Ant却克服了这些工具的缺陷.最初Ant开发者在开发跨平台的应用时,用样也是基于这些缺陷对Ant做了更好的设计. Ant 与 makefileMakefile有一些不足之处,比如很多

转:Java eclipse下 Ant build.xml实例详解

在有eclipse集成环境下ant其实不是很重要,但有些项目需要用到,另外通过eclipse来学习和理解ant是个很好的途径,所以写他demo总结下要点,希望能够帮到大家. 一.本人测试环境eclipse3.6已自动集成了ant环境, 所以就不用单独下载配置ant环境了. 如果没有eclipse集成环境可以自己下载ant http://www.apache.org/ 下载最新的版本解压ant 后设置ANT_HOME, PATH中添加ANT_HOME目录下的bin目录(如:ANT_HOME:D:\

ANT build.xml 编译出错Error running javac.exe compiler

在运行ant脚本时,出现build.xml:97: Unable to find a javac compiler,意思是说java编译时出现了错误,但是我明明我eclipse都能运行啊,后来发现这个eclipse是google自己的eclipse,因此检查系统环境配置时并没有配置jdk的环境,因此首先配置jdk的环境配置,这个不多说,还有就是需要看看javac的路径,Window-->Preferences-->Java-->Installed JRES,如下图 发现只有jre环境,而

android ant build.xml实例

利用ant编译apk. 使用的时候仅仅须要改动SKD路径javahome路径和project路径就能够了. <?xml version="1.0" encoding="UTF-8"?> <project name="Scolview" default="zipalign" basedir="."> <property name="exe" value=&qu

[转]使用ant让Android自动打包的build.xml,自动生成签名的apk文件(支持android4.0以上的版本)

在android4.0以后的sdk里那个脚本就失效了,主要是因为 apkbuilder这个程序不见了: 人家sdk升级,我们的脚本也要跟上趟,修改一下喽. 上网一查,大家的文章还停留在我去年的脚本程度,算了,自己动手查阅了资料之后,具体实现如下: 在工程的根目录 创建2个文件,分别: 1.build.xml 2.build.properties build.xml的内容: [java] view plaincopyprint? <?xml version="1.0" encodi