Ant编译的jar文件,ANE不识别

Ant编译的jar文件,ANE不识别

问题描述

Android的ANE打包需要jar文件。Eclipse可以提供jar文件的导出。

然而,当我使用Ant来自动化完成ANE打包流程的时候,jar文件出了问题。

如果使用Ant生成的jar文件来打包ANE。那么ANE在使用的时候,会发生 ExtensionContext 无法初始化的情况。

也就是说,在调用 ExtensionContext.createExtensionContext(EXTENSION_ID) 的时候,得到的永远是null。

问题分析

以下是构建jar的target:

 1<target name="android" description="Build Android Library">
 2	<delete dir="../android/temp"/>
 3	<mkdir dir="../android/temp/classes"/>
 4	<echo message="Using Java version ${ant.java.version}."/>
 5	<javac source="1.6" srcdir="../android/src" destdir="../android/temp/classes" includeantruntime="false">
 6		<classpath>
 7			<pathelement location="${android.sdk}/android.jar"/>
 8			<pathelement location="${flex.sdk}/lib/android/FlashRuntimeExtensions.jar"/>
 9			<pathelement location="../android/libs/android-support-v4.jar"/>
10			<pathelement location="${android.tools}/support/annotations.jar"/>
11		</classpath>
12	</javac>
13	<mkdir dir="../temp/android/"/>
14	<jar basedir="../android/temp/classes" destfile="../temp/android/lib${name}.jar"/>
15	<copy todir="../temp/android/res/">
16		<fileset dir="../android/res"/>
17	</copy>
18	<delete dir="../android/temp"/>
19</target>

Ant打包得到的jar文件,比使用Eclipse导出的jar文件要小。解压打包的jar文件之后,我发现使用Ant编译出来的class文件比Eclipse编译的要小。

于是,应该从javac上找原因。

解决过程

酱油debug

先为JAVAC task加上 debug="on" 参数。这样编译出来的class文件,大小和Eclipse类似了。但是做成的ANE依然不管用。

JDT编译器

因为Eclipse的编译器和Ant默认使用的编译器是不同的,我尝试将Ant的编译器换成Eclipse看看。使用JDT编译器的步骤为:

  1. 进入 Eclipse downloads 找一个版本,进入下载页面,搜索 JDT Core Batch Compiler,可以下载到JDT编译器;
  2. 将解压后得到的 ecj-<version>.jar 复制到Ant的lib目录;
  3. 为JAVAC task加入 compiler="org.eclipse.jdt.core.JDTCompilerAdapter" 参数,运行task。

JAVAC报告如下错误:

Target level '1.2' is incompatible with source level '1.6'. A target level '1.6' or better is required

为JAVAC task加入 target="1.6" 参数,运行task正常。完整JAVAC参数如下:

1.....
2<javac source="1.6" srcdir="../android/src" destdir="../android/temp/classes" includeantruntime="false" compiler="org.eclipse.jdt.core.JDTCompilerAdapter" target="1.6">
3.....

然后生成ANE进行调试,一切正常,问题解决。

javac编译器

既然使用JDT编译器正常,那么使用javac的默认编译器(JDK自带的编译器)是否也正常?

删除 compiler 参数设置,保留 target 参数设置,运行task正常,生成ANE调试,一切正常。

 1<target name="android" description="Build Android Library">
 2	<delete dir="../android/temp"/>
 3	<mkdir dir="../android/temp/classes"/>
 4	<echo message="Using Java version ${ant.java.version}."/>
 5	<javac source="1.6" target="1.6" srcdir="../android/src" destdir="../android/temp/classes" includeantruntime="false">
 6		<classpath>
 7			<pathelement location="${android.sdk}/android.jar"/>
 8			<pathelement location="${flex.sdk}/lib/android/FlashRuntimeExtensions.jar"/>
 9			<pathelement location="../android/libs/android-support-v4.jar"/>
10			<pathelement location="${android.tools}/support/annotations.jar"/>
11		</classpath>
12	</javac>
13	<mkdir dir="../temp/android/"/>
14	<jar basedir="../android/temp/classes" destfile="../temp/android/lib${name}.jar"/>
15	<copy todir="../temp/android/res/">
16		<fileset dir="../android/res"/>
17	</copy>
18	<delete dir="../android/temp"/>
19</target>

结论

原来,导致class不能使用的真正原因,并非编译器差别,而是在编译的时候没有设置目标版本。Eclipse编译设置中,默认就为编译参数加上了目标版本,而Ant则没有。

我的自动编译ANE功能终于完善了。

参考