Android NDK,坑啊,对于不会c的人来说更是坑啊,弄了一整天才把程序调出来,在网上找了各种资料,各种解决出现的bug,终于是能够运行了。
考虑了一下,还是做个目录吧。
1、创建一个项目
2、编写jni方法
3、生成.h文件
4、创建C++文件,并且复制内容
5、创建Android.mk文件和Application.mk
6、生成so库
先创建一个项目,如下图:
项目创建完成后,可以选择创建一个用来写jni的类,下图是我自己创建的
JNI类.png
创建好了后生成.h文件,过程如下图
用工具生成.h头文件.png
以上使用的Studio自带编写的工具
原文地址
下图是生成之后的
然后新建一个C++文件,取名为jnitest.cpp(自己取名),写上需要include的文件,从.h文件中复制方法过来(方法名、参数类型、返回值等必须一致!) JniTest文件.png
然后创建Android.mk和Application.mk文件(Application.mk可以忽略,简单的jni项目貌似用不到),Android.mk内容如下: Android.mk内容.png
然后生成.so库
ndk-build.png
生成后如下:
生成的so文件.png
之后便可以运行了,结果如图
运行结果.png
这个图片太大了,之截了部分图。
以下说一说期间遇到的一些问题以及解决方案。
<h5>第一个问题</h5>:写第一遍的时候,东西都写好了,运行时说该项目不是jni项目什么的,忘记了[尴尬],找资料找了半天,终于是找到了,在gradle.properties这个文件里加一句话
android.useDeprecatedNdk=true
但是等后来再次测试时,发现有没有无所谓了,也不知道是什么鬼,希望知道的朋友能告诉一声。
<h5>第二个问题</h5>
最后一遍的时候,东西写好了,编译也通过了,但是运行时却出现
java.lang.UnsatisfiedLinkError: Couldn't load JNI_ANDROID_TEST from loader dalvik.system.PathClassLoader[DexPathList[[zip file "/data/app/com.zone.ndktest-1.apk"],nativeLibraryDirectories=[/data/app-lib/com.zone.ndktest-1, /vendor/lib, /system/lib]]]: findLibrary returned null
解决方案:在build.gradle文件的Android中加入
// 告知执行的jni编译.c的文件夹
sourceSets {
main {
jni.srcDirs = ['src/main/jni', 'src/main/jni/']
}
}
// 告知ndk-buid编译所需的Android.mk
externalNativeBuild {
ndkBuild {
path file("src\\main\\jni\\Android.mk")
}
}
这个我也是从配置插件的那个博客中找到的,生成so的两种方式,一种在命令行敲命令,或者配置插件一样(插件以后用的时候方便,不用每次都敲命令了),另一种是在build.gradle里的android中加入以上的代码,然后就可以运行了。 [参考博客](http://blog.csdn.net/langxm2006/article/details/53899638)
<h5>第三个问题</h5>
这个里面的return的内容是我自己添加的,为了输出个“Hello from JNI!”,又花了我好多的时间[大哭],现在这个图片的内容是错误的,截图的时候忘记更改了,没有更改之前运行时会报错的,自己可以试一下
(JNIEnv *,jclass)
上边的内容应该更改为
(JNIEnv * env,jclass o)
env和o是自己起的变量名
至于.h 、.c 、Android.mk和Application.mk文件中的代码的含义,自己表示没看懂。。。问了做iOS开发的朋友,他说他也看不懂,C和OC中都不是那样写的,自己感觉更懵逼了,再写一片博客来记录一下那些代码的含义,以后翻的时候会更方便了。
网友评论