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

相关文章

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

如何在页面调用utility bar并传递参数至lwc组件

1.在app的utility item中添加lwc组件: 2.调用utility bar api的方式有两种: 方法一,通过lwc调用: import {LightningElement,api ,wire } from 'lwc';import { publish, MessageContext } from 'lightning/messageService';import Ca

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

Retrieval-based-Voice-Conversion-WebUI模型构建指南

一、模型介绍 Retrieval-based-Voice-Conversion-WebUI(简称 RVC)模型是一个基于 VITS(Variational Inference with adversarial learning for end-to-end Text-to-Speech)的简单易用的语音转换框架。 具有以下特点 简单易用:RVC 模型通过简单易用的网页界面,使得用户无需深入了

浅谈主机加固,六种有效的主机加固方法

在数字化时代,数据的价值不言而喻,但随之而来的安全威胁也日益严峻。从勒索病毒到内部泄露,企业的数据安全面临着前所未有的挑战。为了应对这些挑战,一种全新的主机加固解决方案应运而生。 MCK主机加固解决方案,采用先进的安全容器中间件技术,构建起一套内核级的纵深立体防护体系。这一体系突破了传统安全防护的局限,即使在管理员权限被恶意利用的情况下,也能确保服务器的安全稳定运行。 普适主机加固措施:

webm怎么转换成mp4?这几种方法超多人在用!

webm怎么转换成mp4?WebM作为一种新兴的视频编码格式,近年来逐渐进入大众视野,其背后承载着诸多优势,但同时也伴随着不容忽视的局限性,首要挑战在于其兼容性边界,尽管WebM已广泛适应于众多网站与软件平台,但在特定应用环境或老旧设备上,其兼容难题依旧凸显,为用户体验带来不便,再者,WebM格式的非普适性也体现在编辑流程上,由于它并非行业内的通用标准,编辑过程中可能会遭遇格式不兼容的障碍,导致操

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验