一.JNI 开发的一般流程
在 windows 系统上面我们经常能看到很多类似于 xxx.dll 的文件,在做 android 开发的时候我们能看到很多 xxx.so 的文件。这些都是啥呢?其实就是用 c 和 c++ 实现生成的动态库,供 windows 和 android 系统来调用。
我们解压 QQ 和支付宝的 apk 找到它的 libs 目录下,会发现有大量的 .so 库文件,还有很多是动态下载加载的我们看不到,能看到的就已经有好几百个了。为什么要这么弄?直接就用 java 开发不行吗?其实好处有很多,比如安全,高效,跨平台等等。今天我们就来看下 JNI 开发的一般开发流程。
1.1 编写 native 方法
public class NdkTest {
public static void main(String[] args) {
NdkTest ndkTest = new NdkTest();
System.out.println("签名密钥:"+ndkTest.getSignaturePassword());
}
// 获取签名密钥
public native String getSignaturePassword();
static{
// 加载某个路径下的动态库
System.load("C:/Users/hcDarren/Desktop/android/NDK/NDK_Day12/x64/Debug/NDK_Day12.dll");
}
}
1.2 生成 xxx.h 头文件
javah -d ../jni -jni com.darren.ndk12.NdkTest
1.3 VS 编写实现方法生成 dll 动态库
// 引入头文件
#include "com_darren_ndk12_NdkTest.h"
JNIEXPORT jstring JNICALL Java_com_darren_ndk12_NdkTest_getSignaturePassword
(JNIEnv *env, jobject jobj){
return (*env)->NewStringUTF(env,"940223");
}
二.详解 .h 头文件和实现文件
/* DO NOT EDIT THIS FILE - it is machine generated */
#include "jni.h" // 引入头 jni.h 文件
/* Header for class com_darren_ndk12_NdkTest */
// 打个标记,防止反复引入 copy 内容
#ifndef _Included_com_darren_ndk12_NdkTest
#define _Included_com_darren_ndk12_NdkTest
#ifdef __cplusplus
// 如果是 c++ 则统一用 C 的编译方式
// 会指示编译器这部分代码按C语言的进行编译,而不是C++的。
// C语言并不支持函数重载,因此编译C语言代码的函数时不会带上函数的参数类型,一般只包括函数名。
extern "C" {
#endif
/*
* Class: com_darren_ndk12_NdkTest
* Method: getSignaturePassword
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_darren_ndk12_NdkTest_getSignaturePassword
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
// 引入头文件
#include "com_darren_ndk12_NdkTest.h"
// JNIEXPORT:在Jni编程中所有本地语言实现Jni接口的一个标志
// jstring:对应 java 中的数据类型 String
// JNICALL:也是一个标记可以去掉,编译运行也不会有问题
// JNIEnv:c 与 java 相互调用的桥梁,它提供了很多函数方法
// jobj:java 传递下来的对象,即上面的 NdkTest
JNIEXPORT jstring JNICALL Java_com_darren_ndk12_NdkTest_getSignaturePassword
(JNIEnv *env, jobject jobj){
// 通过 JNIEnv 把 c 字符串转为 jstring
return (*env)->NewStringUTF(env,"940223");
}
三.JNIEnv 的实现原理
JNI 的基础学习我们主要搞清 JNIEnv 就可以了,只要熟悉它里面的函数方法。但是学习的时候肯定不光要搞清它的方法函数,还需要搞清它的实现原理。如果有留意我之前写的一些文章你会发现 c 和 c++ 有所不同,c++ 是这样的。
JNIEXPORT jstring JNICALL Java_com_darren_ndk12_NdkTest_getSignaturePassword
(JNIEnv *env, jobject jobj){
return env -> NewStringUTF("940223");
}
为什么会有这样的不同?c++ 的实现方式我们后面的文章再去详细讲解,我们先来模拟一下 c 的实现方式:
// 引入头文件
#include <stdio.h>
// 定义结构体指针别名
typedef const struct JNINativeInterface *JNIEnv;
// 定义一个结构体
struct JNINativeInterface{
// 定义的不是函数,函数指针
char*(*NewStringUTF)(JNIEnv*, char*);
};
// 只是做一个模拟
char* Java_com_darren_ndk12_NdkTest_getSignaturePassword(JNIEnv *env){
return (*env)->NewStringUTF(env, "940223");
}
// NewStringUTF 方法的实现
char* NewStringUTF(JNIEnv* jniEnv, char* str){
// 一连串的实现 char* -> jstring
return str;
}
void main(){
// 中间还会有很多的流程,我们能看到的就是调用 Java_com_darren_ndk12_NdkTest_getSignaturePassword
// 模拟 JNIEnv 创建过程
struct JNINativeInterface nativeInterface;
nativeInterface.NewStringUTF = NewStringUTF;
JNIEnv jniEnv = &nativeInterface;// 一级指针
// Java_com_darren_ndk12_NdkTest_getSignaturePassword 参数需要的是 JNIEnv*
JNIEnv* env = &jniEnv;// 虽然只有一个 * ,但是其实他是一个二级指针
char* singnature = Java_com_darren_ndk12_NdkTest_getSignaturePassword(env);
printf("singnature = %s", singnature);
// 然后将 jstring 返回给 java
getchar();
}
视频链接:https://pan.baidu.com/s/1vyxCSn0SWo3-YnoD7Rzryw
视频密码:uqmc