ndk07_JNI訪問java方法,訪問java構造,字符串的轉換

一、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);
    
    }
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,882評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,208評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,746評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,666評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,477評論 6 407
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,960評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,047評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,200評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,726評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,617評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,807評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,327評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,049評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,425評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,674評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,432評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,769評論 2 372

推薦閱讀更多精彩內容