一、JNI調用java中的非靜態方法
- 1.獲取java中聲明的函數的簽名
1>java中在包名為com.hubin.jin 的Text.java文件中申明了getRandomInt()方法
class Text{
//用來觸發JNI執行,演示用的
public native void accessMethod();
int getRandomInt(int max){
return new Random().nextInt(max);
}
}
2>cd到對應的Test.java文件生成的Test.class目錄下執行javap命令
3>再執行 javap -s -p Test(Test 為文件名)命令輸出簽名
- 2.實現C端代碼
JNIEXPORT void JNICALL Java_com.hubin.jin_Test_accessMethod
(JNIEnv *env ,jobject jobj){
//1.得到jclass
jclass jclz = (*env)->GetObjectClass(env, jobj);
//2.獲取jmethidId
//getRandomInt:java方法的名字
//(I)I:方法的簽名
jmethodID mid = (*env)->GetMethodID(env, jclz, "getRandomInt", "(I)I");
//3.調用java getRandomInt()方法 200:getRandomInt的參數
jint random = (*env)->CallIntMethod(env, jobj, mid, 200);
printf("C ramdom:%d\n", random);
}
-
3.動態配置.dll庫
假設使用Visual Studio生成.dll庫之后生成的路徑為: D:\project\JNI_Text\x64\Debug\JNI_text.dll 我們可以將 D:\project\JNI_Text\x64\Debug\ 配置到環境變量 在java工程中就不需要再將.dll庫拷貝到libs目錄下加載,直接loadLibrary即可: static{ System.loadLibrary("JNI_text"); }
二、JNI調用java中的靜態方法
-
1.JVM執行原理
java中訪問靜態方法:類名.方法名 JVM :ClassLoader(類加載器)加載.class文件 如果加載失敗:java.lang.ClassNotFoundException JNI的訪問流程跟虛擬機加載的過程很類似
-
2.創建java中的靜態方法
class Text{ public native void accessStaticMethod(); static String getRandeomUUId(){ return UUID.randomUUID.toString(); } }
-
3.實現C端代碼
//JNI訪問java中的靜態方法 JNIEXPORT void JNICALL Java_com.hubin.jin_Test_accessStaticMethod (JNIEnv *env ,jobject jobj){ //1.找到jclass //原理:通過jobject來搜索class,如果找到了,將這個class 轉變成jclass,然后返回 jclass clz = (*env)->GetObjectClass(env, jobj); //2.找到jmethdId //getRandeomUUId:java方法名字 //()Ljava/lang/String :簽名 jmethodID jmid = (*env)->GetStaticMethodID(env, clz, "getRandeomUUId", "()Ljava/lang/String; "); //3.調用靜態方法,得到java提供的uuid jstring uuid = (*env)->CallStaticObjectMethod(env, clz, jmid); //jstring轉化成char* char * uuid_c = (*env)->GetStringUTFChars(env, uuid, NULL); //生成一個uuid.txt的文本文件到本地 char filename[100]; sprintf(filename, "D://%s.txt", uuid_c); FILE *fp = fopen(filename, "w";);//寫入流 fputs("I Love AV", fp); fclose(fp); printf("文件寫入成功\n"); }
三、JNI訪問java構造方法調用類中的函數
-
1.創建要給觸發執行C語言的函數
public class Test{ //觸發函數 public native Date accessConstructor(); }
-
2.C端方法實現
JNIEXPORT jobject JNICALL Java_com.hubin.jin_Test_accessConstructor (JNIEnv *env ,jobject jobj){ //1.通過類的路徑來從JVM里面找到對應的類 jclass jclz = (*env)->FindClass(env, "java/util/Data"); //2.找到構造函數的jmethodid //<init> :構造方法的名字,所有的構造方法的名字都是他 //()V :簽名 自己用javap命令獲取 jmethodID jmid = (*env)->GetMethodID(env, jclz, "<init>", "()V"); //3.調用newObject實例化一個Date對象,返回值是一個jobject //jni中所有引用的數據類型都會轉化成jobject jobject date_obj = (*env)->NewObject(env, jclz, jmid); //4.獲取Date類中的getTime方法的jmethidID //前提是,我們訪問了相關對象的構造函數創建了這個對象 jmethidID time_mid = (*env)->GetMethodID(env, jclass, "getTime", "()J") //5.執行getTime方法 jlong time =(*env)->CallLongMethod(env, date_obj, time_mid); printf("time: %lld \n", time); return date_obj; }
四、JNI中字符串的轉換
- 1.string在C端的轉換方式一
1>從java到jni到C/C++,編碼的轉換過程
//java內部使用的是utf-16 16bit 的編碼方式
//jni 里面使用的utf-8 unicode編碼方式 英文是1個字節,中文 3個字節
//C/C++ 使用 ascii編碼 ,中文的編碼方式 GB2312編碼 中文 2個字節
2>聲明java本地方法
public native String chineseChars(String str);
3> C端代碼實現
#include <Windows.h>
JNIEXPORT jobject JNICALL Java_com.hubin.jin_Test_chineseChars
(JNIEnv *env ,jobject jobj,jstring in){
jboolean iscp;
char * c_str = (*env)->GetStringChars(env, in, &iscp);
if (iscp == JNI_TRUE){
printf("is copy: JNI_TRUE\n");
}
else if (iscp == JNI_FALSE){
printf("is copy: JNI_FALSE\n");
}
int length = (*env)->GetStringLength(env, in);
const jchar * jcstr = (*env)->GetStringChars(env, in, NULL);
if (jcstr == NULL) {
return NULL;
}
//jchar -> char
char * rtn = (char *)malloc(sizeof(char)* 2 * length + 3);
memset(rtn, 0, sizeof(char)* 2 * length + 3);
int size = 0;
size = WideCharToMultiByte(CP_ACP, 0, (LPCWSTR)jcstr, length, rtn, sizeof(char)* 2 * length + 3, NULL, NULL);
/*if (size <= 0)
{
printf("size: 0 \n", rtn);
return NULL;
}*/
if (rtn != NULL) {
free(rtn);
rtn = NULL;
}
(*env)->ReleaseStringChars(env, in, c_str);// JVM 使用。通知JVM c_str 所指的空間可以釋放了
printf("string: %s\n", rtn);
return NULL;
}
- 2.string在C端的轉換方式二(使用String的構造方法在C端轉換java字符串)
1>java中String 有一個構造函數如下
public String(byte bytes[], String charsetName){
this(bytes,0,bytes.length,charsetName);
}
2>C端調用此構造函數生成字符串
JNIEXPORT jobject JNICALL Java_com.hubin.jin_Test_chineseChars
(JNIEnv *env ,jobject jobj,jstring in){
char *c_str = "馬蓉與寶寶";
//獲取jclass
jclass str_cls = (*env)->FindClass(env, "java/lang/String");
//獲取String構造的jmethodID
//([BLjava/lang/String;)V :簽名
jmethodID jmid = (*env)->GetMethodID(env, str_cls, "<init>", "([BLjava/lang/String;)V");
//將jstring轉換成jbyteArray
jbyteArray bytes = (*env)->NewByteArray(env, strlen(c_str));
//將Char *賦值到bytes
(*env)->SetByteArrayRegion(env, bytes, 0, strlen(c_str), c_str);
jstring charsetName = (*env)->NewStringUTF(env, "GB2312");
return (*env)->NewObject(env, str_cls, jmid, bytes, charsetName);
}