一蚯窥、JNI調(diào)用java中的非靜態(tài)方法
- 1.獲取java中聲明的函數(shù)的簽名
1>java中在包名為com.hubin.jin 的Text.java文件中申明了getRandomInt()方法
class Text{
//用來觸發(fā)JNI執(zhí)行掸鹅,演示用的
public native void accessMethod();
int getRandomInt(int max){
return new Random().nextInt(max);
}
}
2>cd到對應(yīng)的Test.java文件生成的Test.class目錄下執(zhí)行javap命令
3>再執(zhí)行 javap -s -p Test(Test 為文件名)命令輸出簽名
- 2.實現(xiàn)C端代碼
JNIEXPORT void JNICALL Java_com.hubin.jin_Test_accessMethod
(JNIEnv *env ,jobject jobj){
//1.得到j(luò)class
jclass jclz = (*env)->GetObjectClass(env, jobj);
//2.獲取jmethidId
//getRandomInt:java方法的名字
//(I)I:方法的簽名
jmethodID mid = (*env)->GetMethodID(env, jclz, "getRandomInt", "(I)I");
//3.調(diào)用java getRandomInt()方法 200:getRandomInt的參數(shù)
jint random = (*env)->CallIntMethod(env, jobj, mid, 200);
printf("C ramdom:%d\n", random);
}
-
3.動態(tài)配置.dll庫
假設(shè)使用Visual Studio生成.dll庫之后生成的路徑為: D:\project\JNI_Text\x64\Debug\JNI_text.dll 我們可以將 D:\project\JNI_Text\x64\Debug\ 配置到環(huán)境變量 在java工程中就不需要再將.dll庫拷貝到libs目錄下加載,直接loadLibrary即可: static{ System.loadLibrary("JNI_text"); }
二、JNI調(diào)用java中的靜態(tài)方法
-
1.JVM執(zhí)行原理
java中訪問靜態(tài)方法:類名.方法名 JVM :ClassLoader(類加載器)加載.class文件 如果加載失斃乖:java.lang.ClassNotFoundException JNI的訪問流程跟虛擬機加載的過程很類似
-
2.創(chuàng)建java中的靜態(tài)方法
class Text{ public native void accessStaticMethod(); static String getRandeomUUId(){ return UUID.randomUUID.toString(); } }
-
3.實現(xiàn)C端代碼
//JNI訪問java中的靜態(tài)方法 JNIEXPORT void JNICALL Java_com.hubin.jin_Test_accessStaticMethod (JNIEnv *env ,jobject jobj){ //1.找到j(luò)class //原理:通過jobject來搜索class巍沙,如果找到了,將這個class 轉(zhuǎn)變成jclass荷鼠,然后返回 jclass clz = (*env)->GetObjectClass(env, jobj); //2.找到j(luò)methdId //getRandeomUUId:java方法名字 //()Ljava/lang/String :簽名 jmethodID jmid = (*env)->GetStaticMethodID(env, clz, "getRandeomUUId", "()Ljava/lang/String; "); //3.調(diào)用靜態(tài)方法,得到j(luò)ava提供的uuid jstring uuid = (*env)->CallStaticObjectMethod(env, clz, jmid); //jstring轉(zhuǎn)化成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構(gòu)造方法調(diào)用類中的函數(shù)
-
1.創(chuàng)建要給觸發(fā)執(zhí)行C語言的函數(shù)
public class Test{ //觸發(fā)函數(shù) public native Date accessConstructor(); }
-
2.C端方法實現(xiàn)
JNIEXPORT jobject JNICALL Java_com.hubin.jin_Test_accessConstructor (JNIEnv *env ,jobject jobj){ //1.通過類的路徑來從JVM里面找到對應(yīng)的類 jclass jclz = (*env)->FindClass(env, "java/util/Data"); //2.找到構(gòu)造函數(shù)的jmethodid //<init> :構(gòu)造方法的名字,所有的構(gòu)造方法的名字都是他 //()V :簽名 自己用javap命令獲取 jmethodID jmid = (*env)->GetMethodID(env, jclz, "<init>", "()V"); //3.調(diào)用newObject實例化一個Date對象允乐,返回值是一個jobject //jni中所有引用的數(shù)據(jù)類型都會轉(zhuǎn)化成jobject jobject date_obj = (*env)->NewObject(env, jclz, jmid); //4.獲取Date類中的getTime方法的jmethidID //前提是矮嫉,我們訪問了相關(guān)對象的構(gòu)造函數(shù)創(chuàng)建了這個對象 jmethidID time_mid = (*env)->GetMethodID(env, jclass, "getTime", "()J") //5.執(zhí)行g(shù)etTime方法 jlong time =(*env)->CallLongMethod(env, date_obj, time_mid); printf("time: %lld \n", time); return date_obj; }
四削咆、JNI中字符串的轉(zhuǎn)換
- 1.string在C端的轉(zhuǎn)換方式一
1>從java到j(luò)ni到C/C++,編碼的轉(zhuǎn)換過程
//java內(nèi)部使用的是utf-16 16bit 的編碼方式
//jni 里面使用的utf-8 unicode編碼方式 英文是1個字節(jié),中文 3個字節(jié)
//C/C++ 使用 ascii編碼 蠢笋,中文的編碼方式 GB2312編碼 中文 2個字節(jié)
2>聲明java本地方法
public native String chineseChars(String str);
3> C端代碼實現(xiàn)
#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端的轉(zhuǎn)換方式二(使用String的構(gòu)造方法在C端轉(zhuǎn)換java字符串)
1>java中String 有一個構(gòu)造函數(shù)如下
public String(byte bytes[], String charsetName){
this(bytes,0,bytes.length,charsetName);
}
2>C端調(diào)用此構(gòu)造函數(shù)生成字符串
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構(gòu)造的jmethodID
//([BLjava/lang/String;)V :簽名
jmethodID jmid = (*env)->GetMethodID(env, str_cls, "<init>", "([BLjava/lang/String;)V");
//將jstring轉(zhuǎn)換成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);
}