[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.so
或 hellojni.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
对应jint
,long
对应jlong
等。 - 字符串:Java 字符串
jstring
需要转换为 C 字符串char*
。使用GetStringUTFChars
和ReleaseStringUTFChars
进行转换和释放。 - 数组:使用
Get<PrimitiveType>ArrayElements
获取数组元素,使用Release<PrimitiveType>ArrayElements
释放数组元素。 - 对象:使用
GetObjectClass
获取对象的类,使用GetMethodID
获取方法 ID,使用Call<MethodType>Method
调用方法。
4.2 异常处理
- 捕获异常:在本地方法中调用 JNI 函数时,可能会抛出异常。使用
ExceptionCheck
检查是否有未处理的异常。 - 抛出异常:使用
FindClass
和ThrowNew
抛出异常。例如:
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>ArrayElements
和ReleaseStringUTFChars
释放数组和字符串。
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。这里不做过多深入,需要的话可自行去了解。