天天看点

Android6.0 NDK 和 .So 之间的关系

一直有人来问JNI和so库之间的关系,之前很少写博客,也没时间,这里简单的写下,网上也有很多文章,这里阐述了如何编译过去,并且做了编译时需要注意的相关说明。

还有一点需要特别强调,JNI可以理解为辅助ADC,真正要使用的是编译之后的.so,这里使用的是Android6.0,可以忽略。

Android6.0 NDK 和 .So 之间的关系

通过NDK我们可以直接用JNI去动态编译,有2种方法:

我们通过手写的方式写出对应的方法

我们通过javah的命令执行出对应的.h

Android6.0 NDK 和 .So 之间的关系

如何,这里涉及到了Context和Toast,如何直接编译会报错。

这里需要注意2点

javac的是时候要 -cp 带上android.jar的路径

javah的时候要在根目录执行,-classpath带上android.jar的路径

编译通过的效果:

Android6.0 NDK 和 .So 之间的关系

无论是手写还是生成,对应的函数体是不会变的,无法就是在数据结构前加上J,代表着JNI,多了ENV和VM , 看上去也复杂了许多,但是大家不要畏惧,因为开发中有专门的人负责写这一层,上层只需要负责调用就可以了。

这里已经生成了.h文件,我们把它放入JNI文件夹中,Eclipse会自动帮我们编译成 .so 文件,前提是你需要设置好NDK的路径与Builder的设置,只要修改了,就会立即自动编译,当然,我设置的Builder选项就是立即编译的。

Android6.0 NDK 和 .So 之间的关系

默认是生成 armeabi 文件夹,代表了 ARM 的 CPU 。其它应用程序直接使用的话就只需引用这个so,这里的 .so 库是指定了类名和包名,所以使用的时候要注意匹配上,不然会报错。

最后调用下这个So库的方法,输出名字:

Android6.0 NDK 和 .So 之间的关系

当然,你也可以编写.cpp,这样可以用 <code>env-&gt;FindClass("com/immqy/dujinyang/DuJNI");</code> 脱离类名的约束。

版权声明:本文为博主杜锦阳原创文章,转载请注明出处。

继续阅读