java调用c++,使用clion进行JNI开发,ddl包生成以及so包生成
- java基础代码
- 生成C++头部文件
- 使用clion写C++实现代码
- cmke打包构建使用
- java调用C++执行
- linux环境下产生CPP的so包
java基础代码
先写好对应的基础代码,先不管static加载的ddl文件,这里的ddl是后面c++代码打包生成的,使用绝对路径即可
package com.chw.gateway;/*** JNI开发*/
public class JavaWithCppApplication {static {System.load("E:\\document\\CLionProjects\\testDDL2\\cmake-build-debug\\libtestDDL2.dll");}public native int add(int a, int b);public native int sendSty(Student student);public static void main(String[] args) {JavaWithCppApplication obj = new JavaWithCppApplication();int result = obj.add(3, 5);System.out.println("Result: " + result);Student student = new Student("chw", 3);System.out.println("id:" + obj.sendSty(student));}
}
package com.chw.gateway;import lombok.AllArgsConstructor;
import lombok.Data;import java.io.Serializable;@Data
@AllArgsConstructor
public class Student implements Serializable {private String name;private Integer stuId;}
生成C++头部文件
使用javah 生成对应的头部文件,因为c++与java交互的头部文件相对复杂,初学者还是直接生成比较好,后面需要添加方法再自己手动添加,如果类再包里面,就指定报名,如果没有再package里,就直 javah 类名 就可以了
javah com.chw.gateway.JavaWithCppApplication
生成的C++头部文件如下 JavaWithCppApplication.h
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_chw_gateway_JavaWithCppApplication */#ifndef _Included_com_chw_gateway_JavaWithCppApplication
#define _Included_com_chw_gateway_JavaWithCppApplication
#ifdef __cplusplus
extern "C" {
#endif
/** Class: com_chw_gateway_JavaWithCppApplication* Method: add* Signature: (II)I*/
JNIEXPORT int JNICALL Java_com_chw_gateway_JavaWithCppApplication_add(JNIEnv *, jobject, jint, jint);JNIEXPORT int JNICALL Java_com_chw_gateway_JavaWithCppApplication_sendSty(JNIEnv *, jobject, jobject);#ifdef __cplusplus
}
#endif
#endif
使用clion写C++实现代码
clion构建c++依赖库的话选择library进行创建项目
如下引入头文件后,实现我们的方法(JavaWithCppApplication.cpp)。 JavaWithCppApplication.h头文件中引用到了jni.h,这是个java与c++交互的头文件,可以到java安装目录下找win版本的可以在 D:\software\Java\jdk1.8.0_221\include\jni.h 和 D:\software\Java\jdk1.8.0_221\include\win32\jni_md.h 可以把这两个文件直接复制到c++项目路径下。
//
// Created by chw on 2024/3/18.
//
#include <iostream>
#include "JavaWithCppApplication.h"JNIEXPORT int JNICALL Java_com_chw_gateway_JavaWithCppApplication_add(JNIEnv *, jobject, jint a, jint b) {return a + b;
}JNIEXPORT int JNICALL Java_com_chw_gateway_JavaWithCppApplication_sendSty(JNIEnv *env, jobject temp, jobject obj) {jclass jcs = env->FindClass("com/chw/gateway/Student");jfieldID fileStuId = env->GetFieldID(jcs, "stuId", "Ljava/lang/Integer;");jobject intObj = env->GetObjectField(obj, fileStuId);jclass intClass = env->FindClass("java/lang/Integer");jmethodID intValue = env->GetMethodID(intClass, "intValue", "()I");jint value = env->CallIntMethod(intObj, intValue);std::cout << "stuId:" << value << std::endl;jfieldID fileName = env->GetFieldID(jcs, "name", "Ljava/lang/String;");jstring classStr = (jstring) env->GetObjectField(obj, fileName);const char *cstr = env->GetStringUTFChars(classStr, 0);std::cout << "cstr:" << cstr << std::endl;return value;
}
使用的是cmake构建的话,那么就直接在CMakeLists.txt 中添加依赖就可以加上jni 依赖
cmake_minimum_required(VERSION 3.27)
project(testDDL2)set(CMAKE_CXX_STANDARD 17)find_package(JNI REQUIRED)include_directories(${JNI_INCLUDE_DIRS})add_library(testDDL2 SHAREDJavaWithCppApplication.cpp)# 链接 JNI 库
target_link_libraries(testDDL2 ${JNI_LIBRARIES})
cmke打包构建使用
在clion中点击构建,就可以打包产生我们需要的ddl文件了
java调用C++执行
如果你使用的是clion默认安装的c++的话,需要将clion默认安装的c++执行环境配置到系统环境变量,保证在命令行执行g++能够成功执行,我的是在 D:\software\JetBrains\CLion 2023.3.1\bin\mingw\bin 需要将这个路径配置在环境path下,否则java调用c++的时候就会出现ddl链接不到等问题,以及不可用等。配置好环境变量后需要重新启动下idea,否则环境变量还是加载不到,如果还是没生效就重启电脑。(我这里win11就是配置了重启才生效的)
如下成功调用
linux环境下产生CPP的so包
保证linux主机上安装有java的linux版本以及C++,cmake
将本地的项目整个上传到linux环境,到项目路径下,我这里是testDDL2,
cd /home/chw/Documents/jni_test/testDDL2
mkdir build
cd build
cmake ..
make
构建完了之后再生成so文件 /home/chw/Documents/jni_test/testDDL2/build/libtestDDL2.so
如果爆出cmake版本不匹配就把CMakeLists.txt 里面的 cmake_minimum_required(VERSION 3.27) 版本调整一下,降低或者升高匹配一下安装的cmake版本。
如果还爆出某些依赖包找不到的问题,报错Could NOT find JNI (missing: JAVA_INCLUDE_PATH JAVA_INCLUDE_PATH2),详细报错如下:
CMake Error at /usr/local/share/cmake-3.23/Modules/FindPackageHandleStandardArgs.cmake:230 (message):Could NOT find JNI (missing: JAVA_INCLUDE_PATH JAVA_INCLUDE_PATH2)
Call Stack (most recent call first):/usr/local/share/cmake-3.23/Modules/FindPackageHandleStandardArgs.cmake:594 (_FPHSA_FAILURE_MESSAGE)/usr/local/share/cmake-3.23/Modules/FindJNI.cmake:382 (FIND_PACKAGE_HANDLE_STANDARD_ARGS)CMakeLists.txt:20 (find_package)
就加一下下面这里配置在CMakeLists.txt
# JAVA_INCLUDE_PATH为jni.h所在路径,一般在jdk目录下的include中
set(JAVA_INCLUDE_PATH /home/chw/software/jdk1.8.0_211/include)
# JAVA_INCLUDE_PATH2为jni_md.h所在路径,一般在jdk目录下的include/linux中
set(JAVA_INCLUDE_PATH2 /home/chw/software/jdk1.8.0_211/include/linux)
set(JAVA_AWT_INCLUDE_PATH /home/chw/software/jdk1.8.0_211/include)
set(JAVA_AWT_LIBRARY /home/chw/software/jdk1.8.0_211/lib)
set(JAVA_JVM_LIBRARY /home/chw/software/jdk1.8.0_211/lib)
在java中引入对应的so包就可以执行了