首先我们来看so文件的来源
1. 自己写.c文件,然后生成so库
2. 引用别人的静态库,或者动态库来生成新的jni调用库。
我们先来看最简单的编写一个jni调用的so库,包含一个获取字符串的方法,通过这个列子,我们主要是了解怎么在android的工程中调用jni的库,以及要注意的问题。
第一步:首先我们来看下我们的工程的目录
jni目录是我们生成库的文件,里面包含了一个.mk编译文件以及一系列的.c文件
libs目录是我们编译jni目录下的mk文件动态生成的。
当我们写好了jni下面的库文件以后,在当前目录下执行ndk-build则会自动生成我们需要调用的.so库文件。
第二步: java里面怎么调用jni的库文件,我们先看代码public class MainActivity extends Activity {
/* A native method that is implemented by the
* 'hello-jni' native library, which is packaged
* with this application.
*/
public native String stringFromJNI();
/* this is used to load the 'hello-jni' library on application
* startup. The library has already been unpacked into
* /data/data/com.example.HelloJni/lib/libhello-jni.so at
* installation time by the package manager.
*/
static {
System.loadLibrary("hello-jni");
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
String str = stringFromJNI();
System.out.println("*****str:" + str);
}
}在java文件中要做到调用jni文件生成的so库,需要做到2步a. 加载库文件static {
System.loadLibrary("hello-jni");}b. 申明库文件里面的方法public native String stringFromJNI();第三步: 编译并执行编译过程如果出现错误:java.lang.UnsatisfiedLinkError stringFromJNI,找不到库文件里面的方法,则个时候一般是我们jni里面申明的外部调用方法出问题了。那我们先来看看jni里面方法的申明:
#include #include /* This is a trivial JNI example where we use a native method* to return a new VM String. See the corresponding Java source* file located at:** apps/samples/hello-jni/project/src/com/example/hellojni/HelloJni.java*/jstringJava_com_example_jintest_MainActivity_stringFromJNI( JNIEnv* env,jobject thiz ){return (*env)->NewStringUTF(env, "Hello from JNI !");}
JNI函数的命名规则: Java代码中的函数声明需要添加native 关键 字;Native的对应函数名要以“Java_”开头,后面依次跟上Java的“package名”、“class名”、“函数名”,中间以下划线“_” 分割,在package名中的“.”也要改为“_”
然后问题来:
为什么jni函数命名一定要加上调用的类的类名了,这样的话不是会存在只在一个类里面能够调用?
答:这个没有办法,规则是这样的,如果要想在其他地方都调用的话,只有把方法声明为本地静态方法了
public static native String stringFromJNI();