Java调用c++的动态链接库正确方式,踩坑选手!

前言,前几个月泛泛的看了一下c++的相关书籍。在看的过程中,做几个例子啥的没啥感觉。但是在整合起来的过程中还是发现了蛮多问题。

1、JNI是什么?需要怎么理解它?

JNI (Java Native Interface,Java本地接口)是一种编程框架 ,使得Java虚拟机中的Java程序可以调用本地应用/或库,也可以被其他程序调用。 本地程序一般是用其它语言(C、C++或汇编语言等)编写的,并且被编译为基于本机硬件和操作系统的程序。
有些事情Java无法处理时,JNI允许程序员用其他编程语言来解决。例如,Java标准库不支持的平台相关功能或者程序库。也用于改造已存在的用其它语言写的程序,供Java程序调用。许多基于JNI的标准库提供了很多功能给程序员使用,例如文件I/O、音频相关的功能。当然,也有各种高性能的程序,以及平台相关的API实现,允许所有Java应用程序安全并且平台独立地使用这些功能。
理解:JNI是一种约束和规范,我们可以通过java调用动态链接库来实现java不好处理的场景。

2、如何在idea和clion中操作使用?

我这边使用的环境是vs2019、idea2021、clion2019。其他的版本目前没尝试,我估计大概都差不多,书写到这里只是做了一个demo。
Demo的场景是,通过java的jni标准库调用c++动态联,获取float或double的二进制数据,并返回String打印!
我这里踩过的坑不少,提前梳理一下:

  1. 使用clion生成了动态联,但是idea上面报错,内容为没有找到哪些数据。
  2. 使用c++ char数据,在答应的过程中始终无法得到相关数据,在java中打印始终为“”。
  3. clion中的toolchains配置
  4. 配置cmakelist错误,始终无法编译

上面的问题足足折磨我了好几天,因为在过程中,我只是需要做个demo,跑完Hello World之后。想到一个场景,就想去证实一下,没想到……,下面我整理一下我的过程:

2.1Java中编写JNI接口

首先需要使用到JNI中的关键字native关键字,类似和java中的interface一样。eg:

public class FloatAndDouble2Binary {
    public native String float2Binary(float v_f);
    public native String double2Binary(double v_d);
    static {
        System.loadLibrary("doubleAndFloat2binary");
    }

    public static void main(String[] args) {
        double d=3.14;
        String s = new FloatAndDouble2Binary().double2Binary(d);
        System.out.println("double:"+s);
        float f=3.14f;
        String sf = new FloatAndDouble2Binary().float2Binary(f);
        System.out.println("float:"+sf);
    }
}

这样我们就写好了接口,但是如何转换成c++的头文件呢?__有两种方式:

2.1.1 通过指令自动生成

进入到源文件的目录,在搜索框中输入cmd,进入控制台:

javac -h . DFUtils.java

然后在目录里面你会发现多了两个文件:一个以.h结尾的文件,一个以.class结尾的二进制文件。
Ps:将.h的文件放入到一个文件夹中,这里提前说过的。做个伏笔

2.1.2 自己编写c++头文件__(c++头文件以.h,源文件则是.cpp)

这里看多了或者写过c++文件的人就知道如何做了。推荐使用指令生成!

3.1 使用Clion导入项目

3.1.1CmakeLists.txt文件的配置

打开Clion工具,点击New CMake Project from Sources,此时会自动生成一个CMakeLists.txt配置的文件。


cmake_minimum_required(VERSION 3.15)  #cmake 兼容最低的版本
project(double) #项目名
set(CMAKE_CXX_STANDARD 14) #cmake 编译的版本
include_directories(.) #include //包括的路径 可以理解为环境变量
include_directories(F:/include)
include_directories(F:/include/win32)
include_directories(F:/include/win32/bridge)
add_library(doubleAndFloat2binary SHARED
        com_company_float_double_class_FloatAndDouble2Binary.cpp
        com_company_float_double_class_FloatAndDouble2Binary.h)
# 查找当前目录下的所有源文件
# 并将名称保存到 DIR_SRCS 变量
aux_source_directory(. DIR_SRCS)
#add_excutable(项目名 待编译的项目.c 待编译的项目.h),与add_library不要同时使用
#否则会出现编译错,又要生成动态链接又要生成可执行的文件,是不行的!

如果使用jni的,你会得到提示jni无法找到的问题。这时候你需要到你自己jdk的配置目录找到include文件夹,其目录为<JAVA_HOME>/include
在CMakeLists.txt 中加入include_directories(path)即可。

3.1.2 安装toolchains

clion的步骤:File-Settings-Build,Execution,Deployment-ToolChains,我这里最开始安装的Cygwin,但是编译出来的时32为的动态连,在64位的机器上会出错。找了N多资料,还是没有找到,最终下载安装了VS。
安装VS的版本也是有要求的,如果在Toolchains里面提示过高过低,你可以降低或者升级vs的版本即可解决
Environment的目录为你安装vs的根目录,clion的Toolchains会自动搜索,其中Architecture,选择你对应cpu的版本,Platforms目前还不知道目的,有知道的朋友可以帮忙填空。Version自动会进行选择,

3.1.3 编写c++代码。

//com_company_float_double_class_FloatAndDouble2Binary.cpp
#include <jni.h>
#include "com_company_float_double_class_FloatAndDouble2Binary.h"
#include <iostream>

char* strChars;
template <typename T>
char* double2Template(T v);
char* value2Binary(char* point, int sof) {
    const int length = sof * 8+1;
    strChars = new char[length];
    int index = 0;
    for (int i = sof - 1; i >= 0; i--) {
        char item = point[i];
        for (int j = 7; j >= 0; j--) {
            strChars[index] = (item >> j & 1)+'0';
            index++;
        }
    }
    strChars[length-1]='\0';
    return strChars;
}

template<typename T>
char* double2Template(T v) {
    int sof = sizeof(v);
    const int lengt = sof * 8 ;
    char* point = (char*)(&v);
    char* arry = value2Binary(point, sof);
    return arry;
}

/*
 * Class:     com_company_float_double_class_FloatAndDouble2Binary
 * Method:    float2Binary
 * Signature: (F)Ljava/lang/String;
 */
JNIEXPORT jstring

JNICALL Java_com_company_float_1double_1class_FloatAndDouble2Binary_float2Binary
        (JNIEnv *env, jobject obj, jfloat f) {
    int byteLen = sizeof(jfloat);
    double2Template(f);
    return env->NewStringUTF(strChars);
}

/*
 * Class:     com_company_float_double_class_FloatAndDouble2Binary
 * Method:    double2Binary
 * Signature: (D)Ljava/lang/String;
 */
JNIEXPORT jstring

JNICALL Java_com_company_float_1double_1class_FloatAndDouble2Binary_double2Binary
        (JNIEnv * env, jobject obj, jdouble d) {
     double2Template(d);
    return env->NewStringUTF(&*strChars);
}

如果按照我上面的全部编写成功,可以使用快捷键ctrl+F9或者使用[Build-Build Project]进行编译。此时会看到一个.dll的文件。

在java中添加引用dll的文件

static{
	System.load("文件名");
}

将文件赋值到<Java_HOME>/bin目录下,直接运行java调用(也可以在idea中导入dll的动态链接库,网上有很多博客可以参考)。

4.1 Java如何直接将float或者double改成二进制呢?

        //使用java 得到float的二进制数
        float f=3.14f;
        int ibyte= Float.floatToIntBits(f);
        StringBuilder sb=new StringBuilder();
        for (int i=31;i>=0;i--){
            sb.append(ibyte>>i&1);
        }
        // 或者是直接遍历
        // Integer.toBinaryString(ibyte);
        System.out.println(sb);

至此Java调用c++动态库就成功了。一个简单的Hello World就此就完结了。

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java调用C++动态库,可以使用JNI(Java Native Interface)实现。JNI允许Java代码调用本地(即C/C++)代码,具体步骤如下: 1. 编写C++代码并生成动态库文件 首先,你需要编写C++代码,并且生成动态库文件。这里假设你已经编写好了一个名为"mylib"的C++库,并且生成了一个名为"libmylib.so"的动态库文件。 2. 编写Java代码 接下来,你需要编写Java代码来调用C++动态库。以下是一个简单的示例: ```java public class MyLibWrapper { static { System.loadLibrary("mylib"); } public native int myFunction(int arg); public static void main(String[] args) { MyLibWrapper wrapper = new MyLibWrapper(); int result = wrapper.myFunction(42); System.out.println("Result = " + result); } } ``` 在上面的代码中,首先使用`System.loadLibrary()`方法加载C++动态库。然后,定义了一个名为`myFunction()`的本地方法,该方法将在C++动态库中实现。最后,在`main()`方法中创建了一个`MyLibWrapper`对象,并调用了`myFunction()`方法。 3. 生成Java本地方法头文件 在编写Java代码后,你需要生成Java本地方法头文件。可以使用`javac`命令来生成头文件,如下所示: ``` javac MyLibWrapper.java javah -jni MyLibWrapper ``` 上面的命令将生成一个名为MyLibWrapper.h的头文件,其中包含了`myFunction()`方法的声明。 4. 实现本地方法 接下来,你需要在C++动态库中实现`myFunction()`方法。以下是一个示例: ```c++ #include "MyLibWrapper.h" JNIEXPORT jint JNICALL Java_MyLibWrapper_myFunction(JNIEnv *env, jobject obj, jint arg) { // your C++ code here return arg * 2; } ``` 在上面的C++代码中,`Java_MyLibWrapper_myFunction()`方法是`myFunction()`方法的本地实现。在该函数中,你可以编写你的C++代码,并返回一个整数结果。 5. 编译并运行 最后,你需要编译C++代码并将其生成为动态库文件。可以使用以下命令来编译: ``` g++ -shared -fPIC MyLibWrapper.cpp -o libmylib.so ``` 这将生成一个名为"libmylib.so"的动态库文件。然后,你可以运行Java代码来调用C++动态库: ``` java MyLibWrapper ``` 这将输出"Result = 84",表明Java代码已经成功调用C++动态库

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值