JNI 學習筆記(二)-- JNI訪問Java中各方法

版權聲明:著作權歸作者所有。商業轉載請聯系作者獲得授權,非商業轉載請注明出處。我的CSDN地址:http://blog.csdn.net/urrjdg
本文CSDN地址:http://blog.csdn.net/urrjdg/article/details/78158959

CSDN 和 簡書 同步更新
看目錄去CSDN

1. JNI 訪問 java 中的 非靜態方法

package com.zeking.jni;

import java.util.Random;


public class JniTest02 {

    public native void accessMethod();
    
    int getRandom(int max){
        return new Random().nextInt(max);
    }
    
    static{
        System.loadLibrary("lsn02");
    }
    
    public static void main(String[] args) {
        JniTest02 jniTest02 = new JniTest02();
        jniTest02.accessMethod();
    }
}
#include "com_zeking_jni_JniTest02.h"

/*
* Class:     com_zeking_jni_JniTest02
* Method:    accessMethod
* Signature: ()V
*/
// JN訪問 java 中的 非靜態方法
JNIEXPORT void JNICALL Java_com_zeking_jni_JniTest02_accessMethod
(JNIEnv * env, jobject  jobj){

    jclass jclz = (*env)->GetObjectClass(env, jobj);

    jmethodID jmid = (*env)->GetMethodID(env, jclz, "getRandom", "(I)I");

    jint jint = (*env)->CallIntMethod(env, jobj, jmid, 200);

    printf("c random: %d\n",jint);

}

2. JNI 訪問Java 中的靜態方法

package com.zeking.jni;

import java.util.Random;
import java.util.UUID;


public class JniTest02 {

    public native void accessStaticMethod(); 
    
    public static String getRandomUUId(){
        return UUID.randomUUID().toString();
    }
    
    static{
        System.loadLibrary("lsn02");
    }
    
    public static void main(String[] args) {
        JniTest02 jniTest02 = new JniTest02();
        jniTest02.accessStaticMethod();
    }   
}
/*
* Class:     com_zeking_jni_JniTest02
* Method:    accessStaticMethod
* Signature: ()V
*/
// JN訪問 java 中的 非靜態方法
JNIEXPORT void JNICALL Java_com_zeking_jni_JniTest02_accessStaticMethod
(JNIEnv *env, jobject jobj){
    
    jclass jclz = (*env)->GetObjectClass(env, jobj);

    jmethodID jmid = (*env)->GetStaticMethodID(env, jclz, "getRandomUUId", "()Ljava/lang/String;");

    jstring uuid = (*env)->CallStaticIntMethod(env, jclz, jmid);

    char * uuid_c = (*env)->GetStringUTFChars(env, uuid, NULL);

    char filename[100];
    sprintf(filename,"D://%s.txt",uuid_c);

    FILE *fp = fopen(filename, "w");

    fputs("I am Zeking", fp);

    fclose(fp);

    printf("文件寫入成功\n");
}

3. JNI 訪問java構造方法

package com.zeking.jni;

import java.util.Date;
import java.util.Random;
import java.util.UUID;


public class JniTest02 {

    static{
        System.loadLibrary("lsn02");
    }
    
    public native Date accessConstructor();
    
    public static void main(String[] args) {
        JniTest02 jniTest02 = new JniTest02();
        jniTest02.accessConstructor();
    }   
}
/*
* Class:     com_zeking_jni_JniTest02
* Method:    accessConstructor
* Signature: ()Ljava/util/Date;
*/
// 訪問構造方法
JNIEXPORT jobject JNICALL Java_com_zeking_jni_JniTest02_accessConstructor
(JNIEnv * env, jobject jobj){
    // 通過類的路徑來從JVM 里面找到對應的類
    jclass jclz = (*env)->FindClass(env, "java/util/Date");

    // jmethodid     <init> 構造方法
    jmethodID jmid = (*env)->GetMethodID(env, jclz, "<init>", "()V");

    // 調用 newObject 實例化Date 對象,返回值是一個jobjcct
    // 為什么是jobject 不是  Date,因為 在jni 里面 數據的類型對應的關系就是將所有的 引用類型全部轉換為jobjct
    jobject date_obj = (*env)->NewObject(env, jclz, jmid);

    // 得到對應對象的方法,前提是,我們訪問了相關對象的構造函數創建了這個對象
    jmethodID time_mid = (*env)->GetMethodID(env, jclz, "getTime", "()J");

    jlong time = (*env)->CallLongMethod(env, date_obj, time_mid);

    printf("time : %lld \n", time);

    return date_obj;
}

4. 中文

這里寫圖片描述
package com.zeking.jni;

import java.util.Date;
import java.util.Random;
import java.util.UUID;


public class JniTest02 {

    public native String chineseChars(String str);
    
    static{
        System.loadLibrary("lsn02");
    }
    
    public native Date accessConstructor();
    
    public static void main(String[] args) {
        JniTest02 jniTest02 = new JniTest02();
        jniTest02.chineseChars("測試中文");
    }   
}
#include "stdafx.h"
#include "com_zeking_jni_JniTest02.h"
#include <Windows.h>
#include <string.h>

// 亂碼
// java使用的是
// utf - 16 16bit  2個字節

// JNI 里面使用的是
// utf - 8   unicode  可變字節的方式   英文 1個字節    ,中文 3個字節

// C C++
// ascii編碼, 中文 的編碼方式  GB2312  編碼,中文 2個 字節
/*
* Class:     com_zeking_jni_JniTest02
* Method:    chineseChars
* Signature: (Ljava/lang/String;)Ljava/lang/String;
*/
// 中文
JNIEXPORT jstring JNICALL Java_com_zeking_jni_JniTest02_chineseChars
(JNIEnv * env, jobject jobj, jstring in){
    // 方法一
    jboolean iscp;
    // char * c_str = (*env)->GetStringChars(env, in, NULL);
    // JVM會重新開辟一個內存然后存儲 in的值
    char * c_str = (*env)->GetStringChars(env, in, &iscp);
    if (iscp == JNI_TRUE) // 傳&iscp 是為了獲取或者得到 它的返回值,判斷是否會開辟一個新的內存(這個新內存放著一樣的stirng值)給 c 來使用
    {
        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;
    // 就是當我們得到的jcstr指針變量所指的內容全部復制到 對應的指針里面來rtn(這個函數只有window環境才有)
    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;
    }*/
    
    printf("string: %s\n", rtn);
    
    if (rtn != NULL) {
        free(rtn);
        rtn = NULL;
    }
    (*env)->ReleaseStringChars(env, in, c_str);// JVM 使用。通知JVM c_str 所指的空間(這個內存是jvm另外開辟的一個內存,所以要釋放)可以釋放了
    
    
    return NULL;

    // 方法二   
    
    // String 類中構造函數,講byte[] 轉為對應編碼的字符串
    //public String(byte bytes[], Charset charset) {
    //    this(bytes, 0, bytes.length, charset);
    //}
    
    //char *c_str = "中文中文中文";
    //jclass str_cls = (*env)->FindClass(env, "java/lang/String");
    //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);

}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容