JNI调用C++方法指南

2024-08-31 19:28
文章标签 c++ 方法 调用 指南 jni

本文主要是介绍JNI调用C++方法指南,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述

JNI(Java Native Interface)用来在Java中调用其他语言编写的代码。本文主要介绍如何在Java中调用C++程序。

测试环境:Ubuntu 20.04.6 LTS, JDK 8, G++ 9.4.0

JNI的入门

Java中提供了native关键字,用于指示方法实现由本地代码来提供。对于C++程序而言,可以将源码打包为静态库动态库。在JNI中通常使用动态库,这样可以避免将Java字节码和本地代码混合在同一个的二进制文件中。

下面通过一个示例来演示JNI的具体用法。

步骤1:定义Java类HelloJNI,其源代码如下:

public class HelloJNI {static {System.loadLibrary("native"); // 加载指定的本地库, 使得可以通过JNI调用库中定义的本地方法}public static void main(String[] args) {new HelloJNI().sayHello("Linux");}private native void sayHello(String name); // 通过native修饰的本地方法, 需要将其链接到本地代码实现
}

步骤2:通过命令javac -h . HelloJNI.java编译Java源文件并自动生成C++头文件,生成的HelloJNI.h的源码如下:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class HelloJNI */#ifndef _Included_HelloJNI
#define _Included_HelloJNI
#ifdef __cplusplus
extern "C" {
#endif
/** Class:     HelloJNI* Method:    sayHello* Signature: (Ljava/lang/String;)V*/
JNIEXPORT void JNICALL Java_HelloJNI_sayHello(JNIEnv *, jobject, jstring);#ifdef __cplusplus
}
#endif
#endif

本地方法声明JNIEXPORT void JNICALL Java_HelloJNI_sayHello(JNIEnv *, jobject, jstring)中包含三个参数:

  • JNIENV *:代表Java环境,通过该指针可以对Java端的代码进行操作
  • jobject:代表该native方法对应的类对象实例。注意,若该native方法有static修饰,则其代表这个native方法对应的类实例
  • jstring:传递的参数,对应Java中的String类型

步骤3:定义头文件同名的文件HelloJNI.cpp,并在其中实现sayHello()方法:

#include "HelloJNI.h"
#include <iostream>JNIEXPORT void JNICALL Java_HelloJNI_sayHello(JJNIEnv* env, jobject thisObject, jstring name){const char* str = env->GetStringUTFChars(name, NULL);std::cout << "Hello " << str << std::endl;
}

步骤4:编译C++代码为动态库:

# 获取.o文件
g++ -c -fPIC -I${JAVA_HOME}/include -I${JAVA_HOME}/include/linux HelloJNI.cpp
# 打包
g++ -shared HelloJNI.o -o libnative.so

步骤5:运行Java代码:

java -Djava.library.path=. HelloJNI# 结果
Hello Linux

说明:-Djava.library.path=.:指定JVM查找本地库的路径为当前目录。

JNI的高级功能

JNI提供了一组特定的类型用于在C++和Java语言中传递数据。

基本数据类型

下表描述了Java基本类型在C++中对应的本地类型。

Java TypeNative TypeDescription
booleanjbooleanunsigned 8 bits
bytejbytesigned 8 bits
charjcharunsigned 16 bits
shortjshortsigned 16 bits
intjintsigned 32 bits
longjlongsigned 64 bits
floatjfloat32 bits
doublejdouble64 bits
voidvoidnot applicable

引用类型

JNI还支持多种应用类型,其对应关系如下:

Java TypeNative TypeDescription
ClassjclassClass类型
ObjectjobjectJava中的对象
Stringjstring字符串
Object[]jobjectArray对象类型数组
boolean[]jbyteArray布尔型数组
char[]jcharArray字符型数组
short[]jshortArray短整型数组
int[]jintArray整形数组
long[]jlongArray长整型数组
float[]jfloatArray浮点型数组
double[]jdoubleArray双浮点型数组

使用示例

Java类源码:

public class TypeTest {static{System.loadLibrary("typetest");}public static void main(String[] args) {TypeTest test = new TypeTest();int[] arr = {3, 1, 5, 9, 2};System.out.println(test.add(4, 5));System.out.println(test.circle(5.0));int[] brr = test.sortArray(arr);for(int num:brr){System.out.print(num + " ");}System.out.println();}public native int add(int a, int b);public native double circle(double radius);public native int[] sortArray(int[] array);
}

C++类头文件:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class TypeTest */#ifndef _Included_TypeTest
#define _Included_TypeTest
#ifdef __cplusplus
extern "C" {
#endif
/** Class:     TypeTest* Method:    add* Signature: (II)I*/
JNIEXPORT jint JNICALL Java_TypeTest_add(JNIEnv *, jobject, jint, jint);/** Class:     TypeTest* Method:    circle* Signature: (D)D*/
JNIEXPORT jdouble JNICALL Java_TypeTest_circle(JNIEnv *, jobject, jdouble);/** Class:     TypeTest* Method:    sortArray* Signature: ([I)[I*/
JNIEXPORT jintArray JNICALL Java_TypeTest_sortArray(JNIEnv *, jobject, jintArray);#ifdef __cplusplus
}
#endif
#endif

C++类方法实现:

#include "TypeTest.h"
#include <algorithm>const double PI = 3.14;JNIEXPORT jint JNICALL Java_TypeTest_add(JNIEnv * env, jobject obj, jint a, jint b){return a + b;
}JNIEXPORT jdouble JNICALL Java_TypeTest_circle(JNIEnv * env, jobject obj, jdouble radius){return PI * radius * radius;
}JNIEXPORT jintArray JNICALL Java_TypeTest_sortArray(JNIEnv * env, jobject obj, jintArray arr){// 获取数组长度jsize length = env->GetArrayLength(arr);// 获取数组元素jint* elements = env->GetIntArrayElements(arr, 0);std::sort(elements, elements + length);// 创建新的 jintArray 并设置排序后的元素jintArray result = env->NewIntArray(length);env->SetIntArrayRegion(result, 0, length, elements);// 释放原始数组env->ReleaseIntArrayElements(arr, elements, 0);return result;
}

编译运行参考JNI入门章节,不作重复赘述。

结语

参考资料:

  • Chapter 3: JNI Types and Data Structures

这篇关于JNI调用C++方法指南的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1124787

相关文章

使用python生成固定格式序号的方法详解

《使用python生成固定格式序号的方法详解》这篇文章主要为大家详细介绍了如何使用python生成固定格式序号,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录生成结果验证完整生成代码扩展说明1. 保存到文本文件2. 转换为jsON格式3. 处理特殊序号格式(如带圈数字)4

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

JDK21对虚拟线程的几种用法实践指南

《JDK21对虚拟线程的几种用法实践指南》虚拟线程是Java中的一种轻量级线程,由JVM管理,特别适合于I/O密集型任务,:本文主要介绍JDK21对虚拟线程的几种用法,文中通过代码介绍的非常详细,... 目录一、参考官方文档二、什么是虚拟线程三、几种用法1、Thread.ofVirtual().start(

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1

Python实现字典转字符串的五种方法

《Python实现字典转字符串的五种方法》本文介绍了在Python中如何将字典数据结构转换为字符串格式的多种方法,首先可以通过内置的str()函数进行简单转换;其次利用ison.dumps()函数能够... 目录1、使用json模块的dumps方法:2、使用str方法:3、使用循环和字符串拼接:4、使用字符

Python版本与package版本兼容性检查方法总结

《Python版本与package版本兼容性检查方法总结》:本文主要介绍Python版本与package版本兼容性检查方法的相关资料,文中提供四种检查方法,分别是pip查询、conda管理、PyP... 目录引言为什么会出现兼容性问题方法一:用 pip 官方命令查询可用版本方法二:conda 管理包环境方法