Android studio3.6的JNI教程之helloworld

jdk環境變量配置:

path中增加下面2個路徑,也就是android studio的路徑,android有自帶的jdk。

E:\Android\Android Studio\jre\bin
E:\Android\Android Studio\bin
新建工程:

一定要選擇Native c++類型,最後要選c++11支持。

SDK設置:

File->Settings

File->Project Structure

首先確定工程的目錄結構,然後嘗試運行一下工程,使用模擬器,確保工程沒問題,

在MainActivity的同級目錄,新建一個

jni_interface.java,然後做一個簡單的實現,

 

 

jstring str2jstring(JNIEnv* env,const char* pat)
{
    //定義java String類 strClass
    jclass strClass = (env)->FindClass("java/lang/String");
    //獲取String(byte[],String)的構造器,用於將本地byte[]數組轉換爲一個新String
    jmethodID ctorID = (env)->GetMethodID(strClass, "<init>", "([BLjava/lang/String;)V");
    //建立byte數組
    jbyteArray bytes = (env)->NewByteArray(strlen(pat));
    //將char* 轉換爲byte數組
    (env)->SetByteArrayRegion(bytes, 0, strlen(pat), (jbyte*)pat);
    // 設置String, 保存語言類型,用於byte數組轉換至String時的參數
    jstring encoding = (env)->NewStringUTF("GB2312");
    //將byte數組轉換爲java String,並輸出
    return (jstring)(env)->NewObject(strClass, ctorID, bytes, encoding);
}


std::string jstring2str(JNIEnv* env, jstring jstr)
{
    char*   rtn   =   NULL;
    jclass   clsstring   =   env->FindClass("java/lang/String");
    jstring   strencode   =   env->NewStringUTF("GB2312");
    jmethodID   mid   =   env->GetMethodID(clsstring,   "getBytes",   "(Ljava/lang/String;)[B");
    jbyteArray   barr=   (jbyteArray)env->CallObjectMethod(jstr,mid,strencode);
    jsize   alen   =   env->GetArrayLength(barr);
    jbyte*   ba   =   env->GetByteArrayElements(barr,JNI_FALSE);
    if(alen   >   0)
    {
        rtn   =   (char*)malloc(alen+1);
        memcpy(rtn,ba,alen);
        rtn[alen]=0;
    }
    env->ReleaseByteArrayElements(barr,ba,0);
    std::string stemp(rtn);
    free(rtn);
    return   stemp;
}
extern "C" JNIEXPORT jstring JNICALL
Java_com_example_jni_1test_MainActivity_stringFromJNI(
        JNIEnv* env,
        jobject /* this */) {
    std::string hello = "Hello from C++";
    return env->NewStringUTF(hello.c_str());
}
extern "C"
JNIEXPORT jstring JNICALL
Java_com_example_jni_1test_jni_1interface_jni_1call(JNIEnv *env, jobject thiz, jstring str1,
                                                    jstring str2, jstring str3) {
    // TODO: implement jni_call()
}

修改MainActivity.java中的onCreate函數,

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
 
        // Example of a call to a native method
        TextView tv = findViewById(R.id.sample_text);
        //tv.setText(stringFromJNI());
        tv.setText("hello 9+10= " + new hello().add(9, 10));
    }
然後,rebuild project,沒有錯誤後,然後run app。

最終程序整體目錄結構,以及運行效果,

 

JNI的整體流程思路:

Java先定義一個類,類中定義一個需要c++來實現的方法.
通過javah生成需要c++實現的.h的c++頭文件
實現.h的c++頭文件中定義的方法
Cmake編譯運行
 
 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章