JNI调用原理
原理就不细说了,其实就是写个库给Java调,可以百度一下Java JNI,下面是HelloWorld代码测试
编写一个本地测试类
java
package com.my.study.cpp_jni;
/**
* 测试Java调用C++库
* <p>使用命令javac -h . NativeTest.java自动生成C头文件</p>
* <p>到jdk中的include\win32下单jni.h和jni_md.h头文件拷过来</p>
* <p>实现cpp文件</p>
* <p>编译</p>
* <p>导入</p>
* <p>配置cpp库文件</p>
* <p>System.load("D:/xxx/java/com/my/study/cpp_jni/MyLibrary.so");</p>
*
*/
public class NativeTest {
static {
/*
// 设置java.library.path路径
System.setProperty("java.library.path", "/path/to/library");
// 通过System.loadLibrary()加载库文件
System.loadLibrary("mylibrary");
*/
System.load("D:/xxx/java/com/my/study/cpp_jni/MyLibrary.dll");
}
private native void print();
public static void main(String[] args) {
new NativeTest().print();
}
}
java文件格式说明
使用JavaC命令生成C语言的头文件.h文件
用命令javac -h . NativeTest.java自动生成C头文件
CMD
javac -h . NativeTest.java
生成文件com_my_study_cpp_jni_NativeTest.h:
cpp
/* DO NOT EDIT THIS FILE - it is machine generated */
#include "jni.h"
/* Header for class com_my_study_cpp_jni_NativeTest */
#ifndef _Included_com_my_study_cpp_jni_NativeTest
#define _Included_com_my_study_cpp_jni_NativeTest
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_my_study_cpp_jni_NativeTest
* Method: print
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_com_my_study_cpp_jni_NativeTest_print
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
com_my_study_cpp_jni_NativeTest.h文件格式说明
拷贝JDK include目录下的jni.h和win32/jni_md.h头文件到相同路径
修改com_my_study_cpp_jni_NativeTest.h文件的#include <jni.h>为#include "jni.h"引入方式,避免查找C/C++库
NativeTest.cpp实现C头文件函数
cpp
#include<iostream>
#include "com_my_study_cpp_jni_NativeTest.h"
using namespace std;
JNIEXPORT void JNICALL Java_com_my_study_cpp_jni_NativeTest_print(JNIEnv *, jobject) {
cout << "Hello Word JNI" << endl;
}
编译C++库文件
编译C源代码并创建本地库
// 编译C语言库
// gcc -ID:\xxx\Java\Java8\jdk\include\ -ID:\xxx\Java\Java8\jdk\include\win32\ NativeTest.c -shared -o MyLibrary.so
// 编译C++库
// g++ -ID:\xxx\Java\Java8\jdk\include\ -ID:\xxx\Java\Java8\jdk\include\win32\ NativeTest.cpp -shared -o MyLibrary.so
// gcc -I 表示将后面的目录表示为第n个头文件的目录
执行测试,报错
解决错误问题,确认gcc、g++版本是32还是64位
这里不如直接把vs装上测试,因为那个无法找到库文件的问题我弄了几个小时,这里不如直接装上vs,装好vs后几分钟就生成好dll库了
我这里使用VS2022生成动态链接库,装好创建一个C++项目
这里是刚才根据对应头文件和cpp实现文件创建的文件结构,文件都是上面的文件,把文件放到对应位置就行了
对了这里的pch.h预编译头去掉一下,因为用不到,要不然一直报错
使用vs生成动态链接库,把生成的代码放到D盘根目录,方便测试,修改一下Java测试代码,这里可以看到我这里使用g++指令生成的动态链接库比使用vs生成的动态链接库小,说明使用指令生成的动态链接库还差了什么东西在里面,我对C++也不熟悉,所以这里使用vs生成了
java
/**
* 测试Java调用C++库
* <p>使用命令javac -h . NativeTest.java自动生成C头文件</p>
* <p>到jdk中的include\win32下单jni.h和jni_md.h头文件拷过来</p>
* <p>实现cpp文件</p>
* <p>编译</p>
* <p>导入</p>
* <p>配置cpp库文件</p>
* <p>System.load("D:/xxx/java/com/my/study/cpp_jni/_013_Cpp_VS_Dynamic_Link_Lib.dll");</p>
* <p>System.loadLibrary("_013_Cpp_VS_Dynamic_Link_Lib");//不加dll</p>
* Can't load IA 32-bit .dll on a AMD 64-bit platform:需要下载64位cpp编译器,不同cpu编译器不同
*/
public class NativeTest {
static {
System.load("D:/_013_Cpp_VS_Dynamic_Link_Lib.dll");
}
private native void print();
public static void main(String[] args) {
new NativeTest().print();
}
}
idea运行测试:
这里可以看到运行了C++编译打印的字符串Hello World JNI,成功了,撒花
总结:
- 创建Java文件,函数用native标记
- 生成.h头文件
- 拷贝jni.h,jni_md.h文件,调整文件引用位置
- 实现头文件
- 生成动态链接库(这里使用VS)
- Java调用