当前位置: 首页 > news >正文

[Java进阶] 调用C和C++代码利器(JNI)

目录

1. 概述

2. 应用场景

2.1 性能优化

2.2 访问硬件资源

2.3 调用现有库

2.4 跨平台开发

3. 实战案例

3.1 案例目标

3.2 步骤

3.3 总结

4. 最佳实践

4.1 数据类型转换

4.2 异常处理

4.3 线程管理

4.4 内存管理

4.5 性能优化

4.6 调试和测试

4.7 代码组织和维护

4.8 安全性

4.9 跨平台开发

4.10 代码混淆和保护

5. 更好用的工具JNA

1. 概述

Java Native Interface(JNI)是Java编程语言的一种特性,它提供了一种机制来连接Java代码和底层编程语言(如C或C++),JNI的设计目的是为了克服Java语言在操作系统和底层硬件资源访问方面的限制。通过JNI,Java程序员可以使用C或C++编写的本地代码,从而访问底层系统资源,实现更高效的性能和更强大的功能。

JNI的实现涉及到Java虚拟机(JVM)和本地代码之间的交互。Java虚拟机负责加载和执行Java代码,而本地代码是由C或C++等编程语言编写的。JNI提供了一组规范,规定了Java代码和本地代码之间的接口和调用规则。

2. 应用场景

Java Native Interface (JNI) 是一种强大的工具,允许 Java 代码与 C/C++ 代码进行交互。以下是 JNI 的一些常见应用场景:

2.1 性能优化

当 Java 应用程序中某些部分需要高性能处理时,可以使用 C/C++ 编写这些部分,然后通过 JNI 调用。例如:

  • 图像处理:复杂的图像处理算法通常在 C/C++ 中实现,因为这些语言在处理大量数据时性能更好。
  • 科学计算:大规模的科学计算任务,如矩阵运算、数值模拟等,可以用 C/C++ 实现,以提高计算速度。

2.2 访问硬件资源

Java 本身对底层硬件的访问能力有限,而 C/C++ 可以直接访问硬件资源。例如:

  • 设备驱动:访问特定的硬件设备,如摄像头、传感器、打印机等。
  • 嵌入式系统:在嵌入式系统中,通常需要直接操作硬件,而 C/C++ 是更合适的选择。

2.3 调用现有库

许多现有的库是用 C/C++ 编写的,通过 JNI 可以直接调用这些库,而无需重新实现。例如:

  • 图形库:如 OpenGL、DirectX 等。
  • 数学库:如 BLAS、LAPACK 等。
  • 加密库:如 OpenSSL 等。

2.4 跨平台开发

虽然 Java 本身就是跨平台的,但在某些情况下,可能需要在不同平台上使用特定的本地代码。通过 JNI,可以编写跨平台的 Java 代码,同时针对不同平台编写特定的本地代码。例如:

  • 多平台支持:在 Windows、Linux 和 macOS 上使用不同的本地库。

3. 实战案例

下面通过一个具体的实战案例来演示如何使用 JNI 实现 Java 代码与 C/C++ 代码的交互。这个案例将展示如何在 Java 中声明和调用一个本地方法,该方法在 C/C++ 中实现,并返回一个字符串。

3.1 案例目标

  • 在 Java 中声明一个本地方法 getStringFromNative
  • 使用 C/C++ 实现这个方法,使其返回一个字符串 "Hello from C++!"。
  • 编译 C/C++ 代码并生成动态链接库。
  • 在 Java 中加载并调用这个本地方法。

3.2 步骤

创建 Java 类

首先,创建一个 Java 类 HelloJNI,并在其中声明一个本地方法 getStringFromNative

public class HelloJNI {// 声明本地方法public native String getStringFromNative();//......
}

生成头文件

使用 javah 工具生成 C 头文件 HelloJNI.h。假设你的 Java 类文件在 src 目录下。

javac  -encoding  utf8  -h src/ src/HelloJNI.java

生成的 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:    getStringFromNative* Signature: ()Ljava/lang/String;*/
JNIEXPORT jstring JNICALL Java_HelloJNI_getStringFromNative(JNIEnv *, jobject);#ifdef __cplusplus
}
#endif
#endif

实现本地方法

创建一个 C 文件 HelloJNI.c,并在其中实现 getStringFromNative 方法。

#include <jni.h>
#include <stdio.h>
#include "HelloJNI.h"JNIEXPORT jstring JNICALL Java_HelloJNI_getStringFromNative(JNIEnv *env, jobject obj) {return (*env)->NewStringUTF(env, "Hello from C++!");
}

编译本地库

编译 C 代码并生成动态链接库。假设你在 Linux 上,可以使用 gcc 编译:

gcc -shared -o libhellojni.so -I/usr/lib/jvm/java-8-openjdk-amd64/include -I/usr/lib/jvm/java-8-openjdk-amd64/include/linux HelloJNI.c

在 Windows 上,可以使用 cl 编译:

cl -I"C:\Program Files\Java\jdk-11.0.1\include" -I"C:\Program Files\Java\jdk-11.0.1\include\win32" -LD HelloJNI.c -Fehellojni.dll

运行 Java 程序

将生成的动态链接库(如 libhellojni.sohellojni.dll)放在 Java 程序的类路径中,或者指定库的路径。然后运行 Java 程序:

java -Djava.library.path=. -cp src HelloJNI

完整代码

Java 代码 (HelloJNI.java)

public class HelloJNI {// 声明本地方法public native String getStringFromNative();// 加载本地库static {System.loadLibrary("hellojni"); // 库名不带前缀(如 lib)和后缀(如 .dll, .so)}public static void main(String[] args) {HelloJNI helloJNI = new HelloJNI();String result = helloJNI.getStringFromNative();System.out.println(result);}
}

C 代码 (HelloJNI.c)

#include <jni.h>
#include <stdio.h>
#include "HelloJNI.h"JNIEXPORT jstring JNICALL Java_HelloJNI_getStringFromNative(JNIEnv *env, jobject obj) {return (*env)->NewStringUTF(env, "Hello from C++!");
}

运行结果

运行 Java 程序后,你应该看到以下输出:

Hello from C++!

3.3 总结

通过这个简单的案例,我们展示了如何使用 JNI 在 Java 中声明和调用本地方法,并在 C/C++ 中实现这些方法。这个过程涉及以下几个关键步骤:

  • 编写Java代码:在Java代码中声明native方法,这些方法将在本地代码中实现。使用native关键字来声明这些方法,并在类中声明一个本地方法的签名。
  • 生成头文件:使用javac编译器和javah工具(在较新版本的JDK中,javah已被移除,可以使用javac -h选项来生成头文件)来编译Java代码,并生成对应的C/C++头文件。这些头文件包含了本地方法的声明和JNI所需的函数原型。
  • 编写本地代码:使用C或C++等编程语言编写本地代码,实现头文件中声明的本地方法。在本地代码中,可以使用JNI提供的API来访问Java对象和调用Java方法。
  • 编译和链接:将本地代码编译成动态链接库(如Windows下的.dll文件或Linux下的.so文件)。这个动态链接库包含了本地方法的实现,并可以被Java虚拟机加载和执行。
  • 加载动态库:在Java代码中,使用System.loadLibrary或System.load方法加载动态库。一旦动态库加载完成,就可以在Java代码中调用本地方法了。

4. 最佳实践

使用 Java Native Interface (JNI) 时,有一些重要的注意事项和最佳实践,这些可以帮助你避免常见的陷阱,提高代码的健壮性和性能。以下是一些关键点:

4.1 数据类型转换

  • 基本数据类型:JNI 提供了从 Java 到 C/C++ 的基本数据类型映射,如 int 对应 jintlong 对应 jlong 等。
  • 字符串:Java 字符串 jstring 需要转换为 C 字符串 char*。使用 GetStringUTFCharsReleaseStringUTFChars 进行转换和释放。
  • 数组:使用 Get<PrimitiveType>ArrayElements 获取数组元素,使用 Release<PrimitiveType>ArrayElements 释放数组元素。
  • 对象:使用 GetObjectClass 获取对象的类,使用 GetMethodID 获取方法 ID,使用 Call<MethodType>Method 调用方法。

4.2 异常处理

  • 捕获异常:在本地方法中调用 JNI 函数时,可能会抛出异常。使用 ExceptionCheck 检查是否有未处理的异常。
  • 抛出异常:使用 FindClassThrowNew 抛出异常。例如:
jclass exceptionClass = (*env)->FindClass(env, "java/lang/IllegalArgumentException");
if (exceptionClass != NULL) {(*env)->ThrowNew(env, exceptionClass, "This is an illegal argument!");
}

4.3 线程管理

  • Attach/Detach:如果在本地代码中创建了新的线程,需要使用 AttachCurrentThread 将线程附加到 JVM,并在退出时使用 DetachCurrentThread 分离线程。
  • 线程安全:确保本地方法在多线程环境中是线程安全的。使用互斥锁等同步机制来保护共享资源。

4.4 内存管理

  • 局部引用:JNI 会自动管理局部引用的生命周期。局部引用在本地方法返回时自动释放。
  • 全局引用:如果需要在多个本地方法之间共享对象,使用 NewGlobalRef 创建全局引用,并在不再需要时使用 DeleteGlobalRef 释放。
  • 释放数组和字符串:使用 Release<PrimitiveType>ArrayElementsReleaseStringUTFChars 释放数组和字符串。

4.5 性能优化

  • 减少 JNI 调用:频繁的 JNI 调用会有较高的开销。尽量减少 JNI 调用次数,将多个操作合并到一次调用中。
  • 批量处理:如果需要处理大量数据,尽量一次性传递和处理,而不是多次调用。
  • 避免不必要的数据转换:尽量减少数据类型之间的转换,特别是对于大数组和字符串。

4.6 调试和测试

  • 日志记录:在本地代码中使用 printf 或其他日志记录工具来帮助调试。
  • 单元测试:编写单元测试来验证本地方法的正确性。可以使用 C/C++ 单元测试框架,如 Google Test。
  • 性能测试:使用性能测试工具来评估 JNI 调用的性能影响。

4.7 代码组织和维护

  • 模块化:将本地代码组织成模块,每个模块负责特定的功能。这样可以提高代码的可维护性和可读性。
  • 文档:编写详细的文档,说明每个本地方法的用途、参数和返回值。
  • 版本控制:使用版本控制系统(如 Git)来管理本地代码和 Java 代码。

4.8 安全性

  • 输入验证:在本地方法中对输入参数进行严格的验证,防止非法输入导致的安全问题。
  • 权限控制:确保本地方法不会执行超出预期的操作,特别是在处理文件和网络时。

4.9 跨平台开发

  • 条件编译:使用条件编译来处理不同平台的差异。例如:
#ifdef _WIN32
// Windows-specific code
#else
// Unix-like systems code
#endif
  • 动态库路径:确保在不同平台上正确设置动态库的路径。

4.10 代码混淆和保护

  • 混淆:在发布应用程序时,使用代码混淆工具(如 ProGuard)来保护 Java 代码。
  • 本地代码保护:本地代码通常比 Java 代码更难反编译,但仍需注意保护敏感信息,如密钥和算法。

5. 更好用的工具JNA

JNA是建立在JNI基础之上的一个框架,它是对JNI的封装和简化。JNA 是一个第三方库,提供了一种更高层次的接口,允许 Java 代码直接调用动态链接库(如 DLL 或 SO 文件)中的函数,而不需要编写额外的 C/C++ 代码。JNA 自动处理数据类型的转换和内存管理。JNA的性能略逊于JNI。这里不做过多深入,需要的话可自行去了解。


http://www.mrgr.cn/news/62163.html

相关文章:

  • 本地调试自定义Maven Plugin步骤
  • [python3]xlrd不支持Excel xlsx文件类型
  • uniapp实现后端数据i18n国际化
  • 第08章 存储管理(二)
  • 国产编辑器EverEdit - 使用技巧:变量重命名的一种简单替代方法
  • MQ消息队列
  • 力扣——二叉树的后序遍历(C语言)
  • 开发中的拓展属性:增强功能与灵活性
  • SQL语言基础
  • ConcurrentSkipListSet和ConcurrentSkipListMap分析以及总结Set
  • 魔法伤害--是谁偷走了我的0
  • NEEP-EN2-2020-Section1
  • 100种算法【Python版】第28篇——扩展欧几里得算法
  • 【Linux】--- 开发工具篇:yum、vim、gcc、g++、gdb、make、makefile
  • highcharts的datalabels标签格式化
  • 【计网】网络协议栈学习总结 --- 浏览器上输入网址域名后点击回车,到底发生了什么?
  • 通过不当变更导致 PostgreSQL 翻车的案例分析与防范
  • Python金色流星雨(完整代码)
  • ubuntu 挂载 新 硬盘 ext3
  • 高等数学(数二专题)
  • 在Windows下通过pip安装Selenium
  • lenovo联想小新 潮7000-14AST(81GE)笔记本原厂Win10系统镜像安装包下载
  • 软件评测第二期
  • 练习LabVIEW第二十七题
  • window11使用wsl2安装Ubuntu22.04
  • 易至狂欢购车季火热开启,EV3青春版打造年轻一代出行新选择