JNI-NDK 在AndroidStudio3.2.1版本集成方法(ndk-build方式)

1,683 阅读3分钟

集成方法

  1. 在java类里写个native方法

        public class Java2CJNI {
            public  native String java2C();
        }
    
  2. javah 生成jni样式的标准头文件

    切换到src/main/java目录下执行

    javah -d ../jni ndkold.study.com.ndkolddemo.Java2CJNI
    

    -d ../jni 指定输出路径为:当前目录的父目录下的jni目录 ndkold.study.com.ndkolddemo.Java2CJNI这是包含native方法的那个类

3.执行完后生成了jni目录及头文件

这个就是java与native方法进行交互的接口

  1. 创建实现头文件的.cpp源文件 接下来要写个c++代码,实现这个jni接口

˙注意这里新建的是c++代码,c++代码对应下面的代码

//引入刚才生成的头文件
#include "ndkold_study_com_ndkolddemo_Java2CJNI.h"

//复制头文件里的要实现的方法名及其参数
JNIEXPORT jstring JNICALL
Java_ndkold_study_com_ndkolddemo_Java2CJNI_java2C(JNIEnv *env, jobject instance) {
//    实现这个方法,返回一个字符串
    return env->NewStringUTF("Hello from C++");
}

你也可以写个.c源文件,其对应代码为

//引入刚才生成的头文件
#include "ndkold_study_com_ndkolddemo_Java2CJNI.h"

//复制头文件里的要实现的方法名及其参数
JNIEXPORT jstring JNICALL
Java_ndkold_study_com_ndkolddemo_Java2CJNI_java2C(JNIEnv *env, jobject instance) {
//    实现这个方法,返回一个字符串
    return (*env)->NewStringUTF(env, "Hello from C");
    //注意这里是(*env),而且需要传递一个参数(env)
}

首次接触jni可忽略下面的解释,知道怎么用就行,以后再慢慢研究

说明:c与c++就这点区别,查看jni.h文件,发现在c里的JNIEnv是结构体指针JNINativeInterface*的别名,所以JNIEnv *env相当于二级指针,现在要调用JNINativeInterface*里的方法,要用(*env)->xxx

在c++里JNIEnv是_JNIEnv的别名,在_JNIEnv内部里有个属性为结构体指针JNINativeInterface*,然后他把所有c里的方法都重新定义了一下,定义方式就是通过JNINativeInterface*调了一遍所有c里的方法,而且把JNINativeInterface*的对象以this方式传递进去了,可见这里的JNIEnv *env是一个一级指针,所以通过env就可以直接调用对应的方法了(有点绕)

  1. rebuild project 报错
FAILURE: Build failed with an exception.

* What went wrong:
Execution failed for task ':app:compileDebugNdk'.
> Error: Your project contains C++ files but it is not using a supported native build system.
  Consider using CMake or ndk-build integration. For more information, go to:
   https://d.android.com/r/studio-ui/add-native-code.html
  Alternatively, you can use the experimental plugin:
   https://developer.android.com/r/tools/experimental-plugin.html

在AndroidStudio3.2.1版本以下,是其他处理方式,见下文 如何解决Your project contains C++ files but it is not using a supported native build system, 但是我用的是3.2.1版本,按照上述处理还是不行的,要参考这篇文章android Studio(3.2.1) NDK配置

  1. rebuild报错后,复制Android.mk文件

找到Android.mk文件,在build/intermediates/ndk/目录里

复制到这里,jni/

  1. moudle.gralde文件配置
defaultConfig {
        ...
        ndk {
            moduleName "Java2C"
            //so文件名,如果这里配置了so文件名字,
            //记得更改Android.mk里的LOCAL_MODULE :字段为 LOCAL_MODULE := Java2C
            abiFilters "armeabi", "armeabi-v7a", "x86" //指定so文件所支持的CPU类型,如果不写的话,会生成所有的CPU类型的so文件
        }
    }
android {
    ...
     externalNativeBuild {
            ndkBuild {
                path "src/main/jni/Android.mk"//指定Android.mk路径
            }
        }
    }
  1. 再次rebuild

至此生成了libJava2C.so文件

9.调用native方法

public class Java2CJNI {
//    先加载so文件,注意这里的libname是Java2C,不是 libJava2C.so
//    这个libname是Android.mk里的 LOCAL_MODULE := Java2C 字段控制的,注意一定要一致的
    static {
        System.loadLibrary("Java2C");
    }

    public native String java2C();
}

至此,java通过jni调用C++ 成功了

  1. 注意

如果你没成功,很可能是这几处不一致

github demo下载

参考

  1. Android NDK开发之从环境搭建到Demo级十步流

    这篇文章在AndroidStudio版本3.2.1上不适用,适用于更早期的版本

  2. android Studio(3.2.1) NDK配置

    这篇文章适用于AndroidStudio版本3.2.1以上

  3. 官方文档

  4. 慕课网视频-Android-NDK入门(Eclipse)

  5. 慕课网视频-Android-NDK进阶(AndroidStudio)

  6. 简书- JNI从入门到放弃(很好的文章)