首先我們來看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 <string.h>
#include <jni.h>/* 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();