1、
在使用NDK的过程中,有一个步骤是根据生成的 .class 文件生成相应的 .h 文件。
网上相关文档给出的方式是使用如下命令:
cd <project dir>
mkdir jni
javah -calsspath bin -d jni <class>
即,进入工程目录,创建一个jni的文件夹。
然后调用javah命令通过bin目录下的 .class 文件生成 .h 文件并放到jni文件夹中。
需要注意的是,这里的 <class> 是以 包名.类名 的形式指定的,不能省略包名。
但是,运行后出现如下错误:
错误:无法访问 <class>
未找到 <class> 的类文件
javadoc: 错误 - 找不到类 <class>。
Error: 未在命令行中指定任何类。请尝试使用 -help。
后来发现,生成的 .class 文件并不是直接放到bin文件夹下的,而是包含一个名为classes的子目录。
所以,把以上命令改为:
javah -calsspath bin/classes -d jni <package>.<class>
2、
运行后,又出现了以下错误:
错误:无法访问 <class>
错误的类文件: <class file>
类文件具有错误的版本 50.0,应为 49.0
请删除该文件或确保该文件位于正确的类路径子目录中。
com.sun.tools.javac.util.Abort
at com.sun.tools.javac.comp.Check.completionError(Check.java:169)
at com.sun.tools.javadoc.DocEnv.loadClass(DocEnv.java:149)
at com.sun.tools.javadoc.RootDocImpl.<init>(RootDocImpl.java:77)
at com.sun.tools.javadoc.JavadocTool.getRootDocImpl(JavadocTool.java:156)
at com.sun.tools.javadoc.Start.parseAndExecute(Start.java:330)
at com.sun.tools.javadoc.Start.begin(Start.java:128)
at com.sun.tools.javadoc.Main.execute(Main.java:66)
at com.sun.tools.javah.Main.main(Main.java:147)
javadoc: 错误 - 致命错误
这个错误是由于jdk版本不一致造成的。
Eclipse创建 .class文件使用的jdk版本是1.6,而运行javah命令使用的是1.5版本。
修改方法,可以设置Eclipse中的jdk版本,也可以修改系统环境对应的java版本。
总之,使各个命令使用的jdk的版本保持一致即可。
3、
当根据 .h 文件编写好 .c 及 Android.mk 文件之后,需要通过NDK命令编译为 .so 文件。
需要将ndk安装目录添加到环境变量中,然后在jni目录下执行命令 ndk-build 命令。
但是,在 r7 版本下会报以下错误:
<ndk>/prebuilt/linux-x86/bin/awk: 无法执行二进制文件
Android NDK: Host 'awk' tool is outdated. Please define HOST_AWK to ...
<ndk>/build/core/init.mk:258: *** Android NDK: Aborting. 。 停止。
这个错误是因为 ndk 包中的awk命令与我当前系统版本不符。
如果系统本身是存在awk命令的话,只需将 <ndk>/prebuilt/linux-x86/bin/awk 删除或者重命名即可。
4、
如果 .so 文件编译成功,会在项目目录下生成文件夹 libs/armeabi,相应的 .so 文件就在该目录下。
打包APK时,会自动将该目录包含进去,保证程序执行时,能够找到相应的库。
但是,我在运行最终生成的APK时,却出错,提示找不到库:
java.lang.UnsatisfiedLinkError: Couldn't load <so>: findLibrary returned null
at java.lang.Runtime.loadLibrary(Runtime.java:429)
at java.lang.System.loadLibrary(System.java:554)
这个错误是因为,库的命名是有规则的。
在Android.mk文件中通过LOCAL_MODULE定义库的名字时,不要加前缀lib和后缀 .so。
在生成 .so 文件时,会自动添加前缀lib和后缀 .so。
在java中调用System.loadLibrary()时,参数与LOCAL_MODULE定义的名字要保持一致。
另外,如果定义LOCAL_MODULE时添加了前缀lib,则在生成 .so 文件时,不会再重复添加前缀lib。
这时候,自动生成的 .so 文件只比定义多了一个后缀 .so。
但是在java中,仍然不要加前缀lib,也就是与LOCAL_MODULE不加前缀时的情形是一样的。
在使用NDK的过程中,有一个步骤是根据生成的 .class 文件生成相应的 .h 文件。
网上相关文档给出的方式是使用如下命令:
cd <project dir>
mkdir jni
javah -calsspath bin -d jni <class>
即,进入工程目录,创建一个jni的文件夹。
然后调用javah命令通过bin目录下的 .class 文件生成 .h 文件并放到jni文件夹中。
需要注意的是,这里的 <class> 是以 包名.类名 的形式指定的,不能省略包名。
但是,运行后出现如下错误:
错误:无法访问 <class>
未找到 <class> 的类文件
javadoc: 错误 - 找不到类 <class>。
Error: 未在命令行中指定任何类。请尝试使用 -help。
后来发现,生成的 .class 文件并不是直接放到bin文件夹下的,而是包含一个名为classes的子目录。
所以,把以上命令改为:
javah -calsspath bin/classes -d jni <package>.<class>
2、
运行后,又出现了以下错误:
错误:无法访问 <class>
错误的类文件: <class file>
类文件具有错误的版本 50.0,应为 49.0
请删除该文件或确保该文件位于正确的类路径子目录中。
com.sun.tools.javac.util.Abort
at com.sun.tools.javac.comp.Check.completionError(Check.java:169)
at com.sun.tools.javadoc.DocEnv.loadClass(DocEnv.java:149)
at com.sun.tools.javadoc.RootDocImpl.<init>(RootDocImpl.java:77)
at com.sun.tools.javadoc.JavadocTool.getRootDocImpl(JavadocTool.java:156)
at com.sun.tools.javadoc.Start.parseAndExecute(Start.java:330)
at com.sun.tools.javadoc.Start.begin(Start.java:128)
at com.sun.tools.javadoc.Main.execute(Main.java:66)
at com.sun.tools.javah.Main.main(Main.java:147)
javadoc: 错误 - 致命错误
这个错误是由于jdk版本不一致造成的。
Eclipse创建 .class文件使用的jdk版本是1.6,而运行javah命令使用的是1.5版本。
修改方法,可以设置Eclipse中的jdk版本,也可以修改系统环境对应的java版本。
总之,使各个命令使用的jdk的版本保持一致即可。
3、
当根据 .h 文件编写好 .c 及 Android.mk 文件之后,需要通过NDK命令编译为 .so 文件。
需要将ndk安装目录添加到环境变量中,然后在jni目录下执行命令 ndk-build 命令。
但是,在 r7 版本下会报以下错误:
<ndk>/prebuilt/linux-x86/bin/awk: 无法执行二进制文件
Android NDK: Host 'awk' tool is outdated. Please define HOST_AWK to ...
<ndk>/build/core/init.mk:258: *** Android NDK: Aborting. 。 停止。
这个错误是因为 ndk 包中的awk命令与我当前系统版本不符。
如果系统本身是存在awk命令的话,只需将 <ndk>/prebuilt/linux-x86/bin/awk 删除或者重命名即可。
4、
如果 .so 文件编译成功,会在项目目录下生成文件夹 libs/armeabi,相应的 .so 文件就在该目录下。
打包APK时,会自动将该目录包含进去,保证程序执行时,能够找到相应的库。
但是,我在运行最终生成的APK时,却出错,提示找不到库:
java.lang.UnsatisfiedLinkError: Couldn't load <so>: findLibrary returned null
at java.lang.Runtime.loadLibrary(Runtime.java:429)
at java.lang.System.loadLibrary(System.java:554)
这个错误是因为,库的命名是有规则的。
在Android.mk文件中通过LOCAL_MODULE定义库的名字时,不要加前缀lib和后缀 .so。
在生成 .so 文件时,会自动添加前缀lib和后缀 .so。
在java中调用System.loadLibrary()时,参数与LOCAL_MODULE定义的名字要保持一致。
另外,如果定义LOCAL_MODULE时添加了前缀lib,则在生成 .so 文件时,不会再重复添加前缀lib。
这时候,自动生成的 .so 文件只比定义多了一个后缀 .so。
但是在java中,仍然不要加前缀lib,也就是与LOCAL_MODULE不加前缀时的情形是一样的。