JNI 基础 - JNIEnv 的实现原理

红橙Darren
• 阅读 1650

一.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

本文转自 https://juejin.cn/post/6844903608324997133,如有侵权,请联系删除。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
刘望舒 刘望舒
3年前
Android深入理解JNI(一)JNI原理与静态、动态注册
Android框架层Android深入理解JNIAndroid框架层本文首发于微信公众号「刘望舒」前言JNI不仅仅在NDK开发中应用,它更是Android系统中Java与Native交互的桥梁,不理解JNI的话,你就只能停留在JavaFramework层。这一个系列我们来一起深入学习JNI。<!more1.JNI概述Android系统按语言来划分的
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
全靠这份Android知识点PDF大全,月薪30K
第一阶段:Android基础知识回顾:回顾Android开发编程,深入理解Android系统原理和层次结构,深入分析Handler源码和原理;回顾Java,C/C,Kotlin、dart在Android开发中必用的语言,熟悉一下几种语言混淆后的特性;回顾AndroidIPC和JNI的底层原理和热更新技术回顾Native开发要点,使用C结
Stella981 Stella981
3年前
Android打包so文件到apk
Android打包so文件到apk在使用Android源码开发app的时候,怎么使JNI生成的so文件打包进Apk呢,如果这样的就可以只提供给客户一个app就搞定了。下面是具体的实施步骤,这只是一个简单的Demo仅供参考,但是任何复杂的业务逻辑使用原理还是相同的1.含义全代码的so打包. a)建立工程,并使用本地调用,java文件如
Stella981 Stella981
3年前
Android So动态加载 优雅实现与原理分析
背景:漫品Android客户端集成适配转换功能(基于目标识别(So库35M)和人脸识别库(5M)),导致apk体积50M左右,为优化客户端体验,决定实现So文件动态加载.!(https://oscimg.oschina.net/oscnet/00d1ff90e4b34869664fef59e3ec3fdd20b.png)点击上方“蓝字”关注我
Easter79 Easter79
3年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Stella981 Stella981
3年前
Android蓝牙连接汽车OBD设备
//设备连接public class BluetoothConnect implements Runnable {    private static final UUID CONNECT_UUID  UUID.fromString("0000110100001000800000805F9B34FB");
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable