AndroidStudio 来编写jni及生成so

一.JNI和NDK介绍

       JNI(Java Native Interface)是方便Java调用C、C++等Native代码所封装的一层接口,相当于一座桥梁。通过JNI可以操作一些Java无法完成的与系统相关的特性,尤其在图像和视频处理中大量用到。

       NDK(Native Development Kit)是Google提供的一套工具,其中一个特性是提供了交叉编译,即C或者C++不是跨平台的,但通过NDK配置生成的动态库却可以兼容各个平台。源码通过NDK编译后可以生成在Android平台上运行的二进制文件.so及bin文件。

二.配置一下ndk环境

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

三.创建流程

1.在/src/main/ 下创建一个JNI Folder,New--->Folder--->JNI Folder,名字自取。
2.在jni目录下,创建实现的c或c++文件,来实现后续h文件中的方法。
3.创建CmakeLists.txt,在app下与build.gradle同级,内容如下:
#标注需要支持的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 定义当前代码库需要依赖的系统或者第三方库文件(可以写多个)
find_library(
        log_lib # 指定要查找的系统库, 给一个名字
        log    # 真正要查找的liblog.so或者liblog.a
)
# target_link_libraries设置最终编译的目标代码库
target_link_libraries(
        Test  # add_library 生成的
        ${log_lib} # find_library 找到的系统库
)
4.在build.gradle的android里面添加以下内容:
externalNativeBuild {
    cmake {
        path "CMakeLists.txt"
    }
}

       必要的话,在defaultConfig内部加入:

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

    }
}
5.编写java文件,在/src/main/java/com/hly/learn/下创建包含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目录下,执行javah -encoding utf8 com.hly.learn.fragments.JniFragement(包含native方法的java类),会生成com_hly_learn_fragments_JniFragment.h文件,文件内容如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include 
/* 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++文件中实现h中的方法,即步骤2,实现如下:

#include
//导入我们创建的头文件
#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.上层可以调用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;
   }

       当进入界面时,TextView会显示"Hello, my name is seven"

8.C/C++函数调用Java函数

       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++实现如下:

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);
    //得到要调用类的方法
    //参数列表:反射类,方法名称,(参数类型)返回类型
    jmethodID  jmeId = env->GetMethodID(clazz, "getFromJava", "()Ljava/lang/String;");
    //调用方法
    jstring result = (jstring)env->CallObjectMethod(job, jmeId);
    return result;
}

       当点击TextView时,会显示setFromJava()返回的字符串,setFromJava()在底层实现会调用java中的方法getFromJava(),从而最终Toast显示"This string is from java"。

你可能感兴趣的:(AndroidStudio 来编写jni及生成so)