AndroidStudio 來編寫jni及生成so

一.JNI和NDK介紹

???????JNI(Java Native Interface)是方便Java調(diào)用C、C++等Native代碼所封裝的一層接口,相當(dāng)于一座橋梁。通過JNI可以操作一些Java無法完成的與系統(tǒng)相關(guān)的特性,尤其在圖像和視頻處理中大量用到。

???????NDK(Native Development Kit)是Google提供的一套工具,其中一個特性是提供了交叉編譯,即C或者C++不是跨平臺的,但通過NDK配置生成的動態(tài)庫卻可以兼容各個平臺。源碼通過NDK編譯后可以生成在Android平臺上運行的二進制文件.so及bin文件。

二.配置一下ndk環(huán)境

???????打開AndroidStudio, File--->Project Structure--->SDK Location--->Android NDK location 下加入ndk的本地路徑。

三.創(chuàng)建流程

1.在/src/main/ 下創(chuàng)建一個JNI Folder,New--->Folder--->JNI Folder,名字自取。
2.在jni目錄下,創(chuàng)建實現(xiàn)的c或c++文件,來實現(xiàn)后續(xù)h文件中的方法。
3.創(chuàng)建CmakeLists.txt,在app下與build.gradle同級,內(nèi)容如下:
#標注需要支持的CMake最小版本
cmake_minimum_required(VERSION 3.4.1)
#add_library 定義需要編譯的代碼庫 名稱, 類型, 包含的源碼
add_library(
        # Sets the name of the library.
        Test//首字母大寫,要不提示錯誤
        # Sets the library as a shared library.
        SHARED
        # source file
        src/main/jni/test.cpp
)
#find_library 定義當(dāng)前代碼庫需要依賴的系統(tǒng)或者第三方庫文件(可以寫多個)
find_library(
        log_lib # 指定要查找的系統(tǒng)庫, 給一個名字
        log    # 真正要查找的liblog.so或者liblog.a
)
# target_link_libraries設(shè)置最終編譯的目標代碼庫
target_link_libraries(
        Test  # add_library 生成的
        ${log_lib} # find_library 找到的系統(tǒng)庫
)
4.在build.gradle的android里面添加以下內(nèi)容:
externalNativeBuild {
    cmake {
        path "CMakeLists.txt"
    }
}

???????必要的話,在defaultConfig內(nèi)部加入:

externalNativeBuild {
    cmake {
        cppFlags ""
        abiFilters "armeabi-v7a", "arm64-v8a", "x86"

    }
}
5.編寫java文件,在/src/main/java/com/hly/learn/下創(chuàng)建包含native方法的類java文件:
public class JniFragment extends Fragment {
    static {
        System.loadLibrary("Test");
    }
    public native String getFromJNI();

    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(getLayoutId(), container, false);
        return view;
    }
}
6.在/src/main/java目錄下,執(zhí)行javah -encoding utf8 com.hly.learn.fragments.JniFragement(包含native方法的java類),會生成com_hly_learn_fragments_JniFragment.h文件,文件內(nèi)容如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_hly_learn_fragments_JniFragment */

#ifndef _Included_com_hly_learn_fragments_JniFragment
#define _Included_com_hly_learn_fragments_JniFragment
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_hly_learn_fragments_JniFragment
 * Method:    getFromJNI
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_hly_learn_fragments_JniFragment_getFromJNI
  (JNIEnv *env, jobject);

#ifdef __cplusplus
}
#endif
#endif

然后把com_hly_learn_fragments_JniFragment.h mv 到 jni目錄下,然后在c或c++文件中實現(xiàn)h中的方法,即步驟2,實現(xiàn)如下:

#include<jni.h>
//導(dǎo)入我們創(chuàng)建的頭文件
#include "com_hly_learn_fragments_JniFragment.h"
JNIEXPORT jstring JNICALL Java_com_hly_learn_fragments_JniFragment_getFromJNI
        (JNIEnv *env, jobject) {
    //返回一個字符串
    return env->NewStringUTF("Hello, my name is seven");
}
7.上層可以調(diào)用native方法了
public class JniFragment extends Fragment {
   static {
       System.loadLibrary("Test");
   }
   public native String getFromJNI();

   @Override
   public void onAttach(Context context) {
       super.onAttach(context);
   }
   @Override
   public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
       View view = inflater.inflate(getLayoutId(), container, false);
       TextView tv = view.findViewById(R.id.tv);
       tv.setText(getFromJNI());//TextView會顯示"Hello, my name is seven"
       return view;
   }

???????當(dāng)進入界面時,TextView會顯示"Hello, my name is seven"

8.C/C++函數(shù)調(diào)用Java函數(shù)

???????java方法如下:

public class JniFragment extends BaseFragment {
    static {
        System.loadLibrary("Test");
    }
    public native String getFromJNI();

    public native String setFromJava();

    public String getFromJava() {
        return "This string is from java";
    }

    @Override
    public int getLayoutId() {
        return R.layout.jni_layout;
    }

    @Override
    public void initData(View view) {
        TextView tv = view.findViewById(R.id.tv);
        tv.setText(getFromJNI());
        tv.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(mContext, setFromJava(), Toast.LENGTH_SHORT).show();
            }
        });
    }
}

???????C/C++實現(xiàn)如下:

JNIEXPORT jstring JNICALL Java_com_hly_learn_fragments_JniFragment_setFromJava
        (JNIEnv * env, jobject ) {
    //通過反射獲取上層的java類
    jclass clazz = env->FindClass("com/hly/learn/fragments/JniFragment");
    //實例化該類
    jobject job = env->AllocObject(clazz);
    //得到要調(diào)用類的方法
    //參數(shù)列表:反射類,方法名稱,(參數(shù)類型)返回類型
    jmethodID  jmeId = env->GetMethodID(clazz, "getFromJava", "()Ljava/lang/String;");
    //調(diào)用方法
    jstring result = (jstring)env->CallObjectMethod(job, jmeId);
    return result;
}

???????當(dāng)點擊TextView時,會顯示setFromJava()返回的字符串,setFromJava()在底層實現(xiàn)會調(diào)用java中的方法getFromJava(),從而最終Toast顯示"This string is from java"。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

友情鏈接更多精彩內(nèi)容