1. Java调用本地代码常见的两种方案
JNI(Java Native Interface),有过不同语言间通信开发经历的一般都知道,它允许java和其他语言代码(尤其是C/C++)进行交互,只要遵守约定即可。首先看下JNI调用C/C++过程,注意写程序时自下而上,调用时自上而下:
可见步骤之多,调用.dll/.so共享库之痛苦的过程。
若已有编译好的.dll/.so文件 —> 需先用是C语言另外写一个.dll/.so共享库,使用SUN规定的数据结构代替C语言的数据结构,调用已有的dll/so中公布的函数 —> java中载入这个库 —> java编写Native函数作为链接库中函数的代理
问题是很少有java程序员愿意编写调用.dll/.so库中原生函数的java程序,这也使java在客户端上乏善可言,是JNI的一大弱点!
JNA提供一组Java工具类用于在运行期间动态访问系统本地库(native library:如Window的dll)而不需要编写任何Native/JNI代码。开发人员只要在一个java接口中描述目标native library的函数与结构,JNA将自动实现Java接口到native function的映射。
简而言之,就是jna基于jni的方式封装了很多api,在使用上面相对于jni来说简化了很多。
但是JNA不能完全替代JNI,JNI不仅可以实现java访问C,也可实现C调用java。
而JNA只能实现Java访问C函数,作为一个Java框架,自然不能实现C语言调用Java代码。此时,有些情况还是需要使用JNI技术。
2. 安装VS2022
下载 VS2022 时,现在社区版即可。
https://visualstudio.microsoft.com/zh-hans/vs/
下载下来是一个在线安装的安装器VisualStudioSetup.exe
,双击运行即可。安装选择 使用C++的桌面开发 即可:
3. 创建java项目
编写本地方法
public class JniDemo {
// 声明本地方法
public native int add(int a, int b);
public native int sub(int a, int b);
}
使用javah
生成c/c++头文件
javah com.jnidemo.JniDemo
生成的头文件com_jnidemo_JniDemo.h
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_jnidemo_JniDemo */
#ifndef _Included_com_jnidemo_JniDemo
#define _Included_com_jnidemo_JniDemo
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_jnidemo_JniDemo
* Method: add
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_com_jnidemo_JniDemo_add
(JNIEnv *, jobject, jint, jint);
/*
* Class: com_jnidemo_JniDemo
* Method: sub
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_com_jnidemo_JniDemo_sub
(JNIEnv *, jobject, jint, jint);
/*
* Class: com_jnidemo_JniDemo
* Method: acl
* Signature: ()D
*/
JNIEXPORT jdouble JNICALL Java_com_jnidemo_JniDemo_acl
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
4. 创建c/c++项目
打开VS2022,创建动态链接库dll项目
输入项目名和存储的目录,点击创建即可
把头文件com_jnidemo_JniDemo.h
拷贝到JniTestDemo项目中,并附加到项目里
创建com_jnidemo_JniDemo.cpp
源文件,并实现头文件的两个定义的方法
#include "pch.h"
#include <jni.h>
#include "com_jnidemo_JniDemo.h"
/*
* Class: com_jnidemo_JniDemo
* Method : add
* Signature : (II)I
*/
JNIEXPORT jint JNICALL Java_com_jnidemo_JniDemo_add(JNIEnv * env, jobject jobj, jint a, jint b){
return a + b;
}
/*
* Class: com_jnidemo_JniDemo
* Method: sub
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_com_jnidemo_JniDemo_sub(JNIEnv* env, jobject jobj, jint a, jint b){
return a - b;
}
添加jdk的jni.h文件到项目中,进行依赖构建
右击项目,现在属性
选择图中这两个目录,做为构建依赖目录;这两个目录分别存在jni.h
和jni_md.h
两个头文件
构建c/c++项目
另外要特别注意编译的dll的x86与x64一定要与JDK的版本对应,否则在java调用dll时会出现以下错误:
Can't load IA 32-bit .dll on a AMD 64-bit platform
5.编写java代码调用dll
package com.jnidemo;
public class JniDemo {
// 声明本地方法
public native int add(int a, int b);
public native int sub(int a, int b);
static {
System.out.println(System.getProperty("java.library.path"));
// x64
System.load("E:\\dllws\\JniTestDemo\\x64\\Debug\\JniTestDemo.dll");
}
public static void main(String[] args) {
JniDemo ob = new JniDemo();
System.out.println(ob.add(1, 2));
System.out.println(ob.sub(2, 3));
}
}
6. 加载dll&so库文件的路径java.library.path说明
- Java的
System.load
和System.loadLibrary
都可以用来加载库文件。例如可以这样载入一个windows平台下JNI库文件:
System.load("D://dll//TestJNI.dll"); //绝对路径
-
System.loadLibrary
参数为库文件名
例如可以这样载入一个windows平台下JNI库文件
System.loadLibrary ("TestJNI");
这里TestJNI必须在java.library.path
这一jvm变量所指向的路径中,可以通过如下方法获得该变量的值:
System.getProperty("java.library.path");
默认情况下,Windows平台下包含下面的路径:
1)和jre相关的目录
2)程序当前目录
3)Windows目录
4)系统目录(system32)
5)系统环境变量path指定的目录
- 在linux下添加一个
java.library.path
的方法如下:
在/etc/profile
后面加上一行
export LB_LIBRARY_PATH=路径
- 在执行程序的时候可以显示指定,
-Djava.library.path=路径
,这种会清除掉预设置的java.library.path
的值。
java -jar -Djava.library.path=/usr/lib TestJni.jar