美文网首页
AndroidStudio 来编写jni及生成so

AndroidStudio 来编写jni及生成so

作者: 雷涛赛文 | 来源:发表于2020-08-31 16:30 被阅读0次

    一.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 <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++文件中实现h中的方法,即步骤2,实现如下:

    #include<jni.h>
    //导入我们创建的头文件
    #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

          本文链接:https://www.haomeiwen.com/subject/wmwusktx.html