JAVA如何调用C/C++动态库

article/2025/10/16 22:15:22

一、调用方式:

JAVA调用C/C++动态库有很多方法,常用的有JNI(Java Native Interface)、JNA(Java Native Access)。

  • JNI:早在JAVA1.1版本就开始支持,它定义了一种公用的语法,当java和c/c++双方都遵循该语法时,可以互相调用。所以使用JNI不能直接调用一般的C/C++库,而必须借助于一个中间动态库,该中间动态库实现了JAVA-JNI语法-C/C++的转换(或者你所调用的动态库原生就封装了JNI)。如果对C++稍微懂一点,其实使用起来也不难。
  • 作者:宋清日
    链接:https://zhuanlan.zhihu.com/p/465601205
    来源:知乎
    著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
     

    java通过JNI(Java Native Interface)与其他语言编写的代码进行交互。

    JNI工作示意图(网上下载的)

    Java要调用第三方动态库,通俗点说就是需要将这个第三方动态库按照Java语言的要求再封装一次,变成Java可以调用的新动态库,这个新动态库去调用原始的动态库。

  • 编写带有native声明的方法的Java类,该方法要与真正调用的动态库的方法和参数和返回值均一致。(直接用IDEA新建Java项目)
  • package com.JniDemo;public class JniDemo {static {System.load("/root/Jni_Lib/libJniDemo.so");}public native int add(int a, int b);public native String print(String msg);public static void main(String[] args){JniDemo demo = new JniDemo();demo.print("11");}
    }

    2. 编译Java类生成.class文件。(build一下创建的project)

    3. 使用javah生成JNI头文件。

    每次头文件有改动的话,直接用工具重新生成,比较方便。

    4. 拿到生成的头文件。

    /* 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:    print* Signature: (Ljava/lang/String;)Ljava/lang/String;*/
    JNIEXPORT jstring JNICALL Java_com_JniDemo_JniDemo_print(JNIEnv *, jobject, jstring);#ifdef __cplusplus
    }
    #endif
    #endif
    

    以上就完成JNI头文件的生成,在小编看来上面这些步骤Linux与window都一样,都是生成了Java可以调用的C++头文件,至于后面调用window的dll库或者Linux上的so库,才会有区别。

    5. 新建C++动态库项目(Linux)

    新建的动态库项目首先要包含三个文件,首先就是生成的JNI头文件com_JniDemo_JniDemo.h,另外两个是jni.h(在JDK目录的include目录下,/usr/java/jdk1.8.0201/include/jni.h)和 jni_md.h(在JDK目录的include的linux目录下,/usr/java/jdk1.8.0_201/include/linux/jni_md.h)。

    后面两个文件可以直接复制到你的动态库项目里面,不用再配置文件路径了,比较方便,另外一个小细节可以注意一下,com_JniDemo_JniDemo.h文件中包含jni.h头文件的时候用的是#include <jni.h>, 如果把文件考到项目中,则需要改成#include "jni.h"。

    6. 新建JniDemo.cpp文件,编译生成动态库。

    #include "com_JniDemo_JniDemo.h"
    #include <iostream>
    #include <string>using namespace std;JNIEXPORT jint JNICALL Java_com_JniDemo_JniDemo_add(JNIEnv *env, jobject job, jint a, jint b)
    {jint c;c = a + b;return c;
    }JNIEXPORT jstring JNICALL Java_com_JniDemo_JniDemo_print(JNIEnv * env, jobject job, jstring s)
    {char str[] = "welcome";std::string hello = "hello form c++";cout << hello << endl;return env->NewStringUTF(hello.c_str());
    }

    7. 将生成的动态库放到第一步指定的路径下。(/root/Jni_Lib/libJniDemo.so)

    8. 运行Java程序。

File &amp;amp;gt;&amp;amp;gt; Setting... &amp;amp;gt;&amp;amp;gt; Tools &amp;amp;gt;&amp;amp;gt; External Tools新建一个工具
  • JNA:在C#中,使用[DLLImport]可以非常方便的调用原生的C/C++动态库,所以sun公司开发了JNA来使JAVA可以像C#一样方便的调用动态库。不过在使用过程中感觉内部原理还是使用了JNI的语法库。不过至少对于我们不懂C++的来说,可以直接调用原生的动态库,而不需要再去生成一个C++的中间动态库了。

所以这次在尝试了JNI之后还是选择了JNA。

</br>

二、关于x32和x64:

如果C/C++动态库使用x32,那必须运行在x32的Tomcat上,并且使用x32的JRE。所以如果不幸拿到股东动态库,未提供x64版本,那只能单独部署x32服务器提供对应的接口,而x64主站点通过http方式调用该接口程序来访问动态库。

如果在x64Tomcat上加载x32版本动态库,将得到如下错误信息:

"Can't load IA 32-bit .dll on a AMD 64-bit platform"

调用方法及DLL存放位置:

先来简单的看一下JNI和JNA两种方式加载动态库的代码:

JNI:

public class ImportDllTest {//加载动态库static{//通过决定地址加载动态库//System.load("d:\\C2JavaTest.dll");//通过库名加载动态库,不加.dll后缀,自适应.dll和liunx平台的.soSystem.loadLibrary("C2JavaTest");}//定义动态库接口方法public native String subString(String str, int startIndex, int length);public static void main(String[] args){ImportDllTest importDll = new ImportDllTest();String str = importDll.subString("test",1,2);}
}

JNA:

引用JNA包

<dependency><groupId>com.sun.jna</groupId><artifactId>jna</artifactId><version>3.0.9</version>
</dependency>

定义接口类,继承com.sun.jna.Library

//必须继承com.sun.jna.Library
public interface ImportDllTest extends Library {//加载动态库,使用动态库名称加载,不带.dll后缀,会自动识别liunx环境下的.so库。也可以使用决定地址加载动态库public static ImportDllTest Instance = (ImportDllTest) Native.loadLibrary("C2JavaTest", ImportDllTest.class);//定义动态库接口方法String subString(String str, int startIndex, int length);public class Main{public static void main(String[] args){String str = ImportDllTest.Instance.subString("test",1,2);}}
}

关于是否加载到动态库:

两种加载动态库都可以使用动态库名称或者绝对路径来加载,在调试过程中,JNA如果没找到动态库并不会给出明确的提示,而JNI的加载方法会明确抛出找不到动态库的异常,所以即使是使用JNA方式,但在一开始不确定是否将动态库放在了正确位置,是否成功的加载了动态库的时候可以借助于JNI的Systetm.loadLibrary来协助判断是否成功的加载了对应的动态库。
</br>

三、动态库位置:

加载动态库都可以使用决定地址来加载。但本次项目加载的动态库会需要调用同级目录下的配置文件,使用绝对地址的方式我是没能成功的加载到配置文件(因为也不清楚动态库里实际加载配置文件的具体实现)。使用动态库名称加载,那动态库到底应该放在哪里,网上查了很多资料,有放system32的,有放jdk/jre的,有放tomcat里的,有放环境变量配置里的。在追求尽量不给后续运维造成太大困扰(放系统system32,或jdk/jre,部署时很容易忘记或错乱),尽可能找最优的方案。
最后我借助于动态库生成的日志文件,来判断默认加载动态库的路径:

应用程序:

Main方法测试时,动态库及配置文件需要放到运行时选择的工作目录下。

动态库和配置文件要放在:

同时生成的日志文件也将会在该目录下

Tomcat部署程序

Tomcat部署时,尝试过很多方式,但是最后选择了Tomcat的bin目录

cd Tomcat/bin

</br>

四、JAVA与C/C++参数对应

java的char是2字节,byte是1字节;c/c++的char是1字节;

作为JAVA传入C/C++参数:

JAVAC/C++
byte[]char[]/char*
Stringchar[]/char*
intint

作为JAVA中传入,C/C++里out的参数:

C/C++JAVA
char[]/char*byte[]

调用是要先将定义的byte数组空间定义好,c++中才可以在已经定义的空间中写值。

//out参数长度8字节内容
byte[] out_param = new byte[8];

</br>

五、总结

  • 可以借助于JIN确定动态库是否能正确加载到。如果动态库不需要配置文件,完全可以使用决定路径来进行加载。
  • 明确动态库的版本,是x32还是x64。不同版本要使用对应的tomcat和jre。
  • 确定传递参数类型,java中一定要记得不可以使用char来和c++的char交互,一定是要byte或String(为什么String可以,猜测可能JIN或JNA在内部转换成了byte)。
  • 最后如果动态库能有人员配合一起调试,那是一个美好的事情。
    </br>
    </br>

这次把成功的几个关键点调用整理在此。本次项目也是因为调用的古董动态库,没有文档,没有错误说明,所有返回都靠猜测,所以不确实是java调用问题还是本身业务问题。前前后后折腾好几周,最后总算还是有了一个好的结果。

以上有任何不对的地方,敬请指正。


http://chatgpt.dhexx.cn/article/5wsXFkab.shtml

相关文章

java调用C++的过程?

jni是java和C、C通信的桥梁。 java适合写上层的应用&#xff0c;C、C适合写底层的应用。因为C、C就是跟底层打交道的。 当然这里并不讨论那么多&#xff0c;只是我要记住这一点&#xff0c;我所以做的工作就是如何使用jni&#xff0c;把留在.java中的本地的接口&#xff0c;让C…

JAVA调用C语言程序

JAVA调用C语言程序 JAVA调用C语言程序1. 编写带有native声明的方法的Java类2. 使用javah 生成&#xff1a;jniSample.h的头文件3. 使用C实现本地sum方法&#xff1a;(这里我生成文件时候&#xff0c;误写了jinSample)。4. 将本地方法编写的文件生成动态链接库。5. 再次刷新项目…

Java简单调用C语言函数

Java简单调用C语言函数 1.安装Dev-Cpp2.编写并编译Java代码3.新建工程4.修改C代码4.1将第2步生成的.h文件复制黏贴到dll.h里面4.2修改dllmain.cpp文件4.3 保存并编译4.4编译错误解决 5.将编译好的dll文件放到java安装目录下的bin目录下6.运行 1.安装Dev-Cpp 2.编写并编译Java代…

C 调用 Java 方法

文章目录 1 Java 代码2 C 代码3 总结3.1 获取参数3.2 调用方法 1 Java 代码 JNI 代码 package com.karashok;import java.util.UUID;public class JNIDemo {/*** 获取静态方法返回值*/public static native String sayHello();/*** 获取方法返回值*/public native String say…

java调用c/c++代码

JNI是Java Native Interface的英文缩写, 中文翻译为本地调用, 自从Java 1.1开始就成为了Java标准的一部分。 C/C是系统级的编程语言, 可以用来开发任何和系统相关的程序和类库, 但是Java本身编写底层的应用比较难实现, 使用JNI可以调用现有的本地库, 极大地灵活了Java的开发。…

java调用C++代码

首先我的参考博客如下&#xff1a; https://www.cnblogs.com/CLAYJJ/p/7725975.html https://www.cnblogs.com/xiaocainiao2hao/p/5619862.html https://www.cnblogs.com/langtianya/p/3470896.html 流程如下&#xff1a; 1.建立一个java文件&#xff0c;在这里我只写了一…

C语言调用Java JNI

最近项目中需要使用JNI&#xff0c;所以研究了一下&#xff0c;其中遇到过不少问题&#xff0c;总结一下&#xff0c;让遇到同样问题的人可以得到解决。 在C/C中调用Java的方法一般分为五个步骤&#xff1a;初始化虚拟机、获取类、获取类的方法、创建类对象、调用方法和退出虚…

java程序如何调用C++代码

看到java多线程中的Thread.isAlive()的类型为native&#xff0c;进一步去联想native方法什么时候用呢&#xff1f;自己能不能编写native方法&#xff1f; 经网上查资料&#xff1a; java中native修饰符的含义为“a native method is a java method whose implementation is p…

Java调用C++程序实现方法

Java调用C程序实现方法(1) 这篇博文是自己在学习过程中&#xff0c;踩了很多坑之后&#xff0c;为了避免广大热爱编程的盆友再像我这样浪费太多时间&#xff0c;索性将自己的实现方法总结出来&#xff0c;以供大家参考。程序最终实现的是Hello World效果&#xff0c;话不多说&…

从 C/C++ 程序调用 Java 代码

JNI允许您从本机代码内调用 Java 类方法。 要做到这一点&#xff0c;通常必须使用 Invocation API 在本机代码内创建和初始化一个 JVM。 下列是您可能决定从 C/C 代码调用Java 代码的典型情况&#xff1a; 1.希望实现的这部分代码是平台无关的&#xff0c;它将用于跨多种平台…

使用Java调用C/C++

文章目录 前言JNI概述例子编写Java代码编译生成的class文件:JNITest.class在命令行下使用javah生成C/C头文件。在工程的bin目录下输入以下命令&#xff1a;实现C代码。在VS2008中创建一个Win32 project&#xff0c;类型为DLL。构建C项目将dll文件复制到Java工程的bin目录下 前言…

java调用c/c++

最近项目想将比较重要的配置文件加密、综合考虑后决定用java jni实现&#xff0c;步骤如下 1.定义java本地接口 package com.msg.jni;public class JniMsg {static {try {String os System.getProperty("os.name").toLowerCase();String path "F:/opt/"…

C/C++如何调用Java

前言 简单介绍C/C如何调用Java&#xff0c;内容适合未接触过此类工程的朋友作为上手参考。 一、编译环境 1. 64位的win10系统 2. JDK&#xff1a;jdk-8u181-windows-x64.exe 3. IDE VS2017 二、调用步骤及Java虚拟机使用方法 一般步骤&#xff1a; 编写Java代码, 并编译…

java调用C

文章目录 idea创建一个java类通过命令生成.h文件vs2015创建dll项目修改dll项目的属性补充dll项目的头文件和源文件导入文件源文件的制作 生成dll文件idea导入dll文件idea使用函数 这个确实卡了我一段时间。 先说说必要性吧。 java无法处理的操作&#xff08;指向地址&#xff…

Java调用c/c++(JNI)最详细步骤

一、JNI(Java Native Interface)的作用就是Java通过JNI调用其他语言的函数(或方法)&#xff08;主要是C&C&#xff09;。 二、准备 1. java8系列jdk&#xff0c;有很多版本&#xff0c;任选一个即可&#xff0c;如jdk1.8.0.231。安装好&#xff0c;配置好环境。 2. vs20…

使用java调用C语言程序教程

1.idea创建一个java类 严格来说&#xff0c;核心步骤并不是创建一个java类&#xff0c;而是创建一个方法&#xff0c;那个方法要被native修饰&#xff0c;这才是关键。 接着在TestNativeCode类当中声明我们的本地方法&#xff1a; package com.wwj.nativecode; public class T…

linux中文语言包下载地址,centos中文语言包-官方版-centos中文语言包fonts-chinese-3.02-12.el5.noarch.rpm-独木成林...

centos中文语言包 fonts-chinese-3.02-12.el5.noarch.rpm&#xff0c;直接在目录下运行&#xff1a; rpm -ivh fonts-chinese-3.02-12.el5.noarch.rpm linux中文文件名乱码的解决办法(安装中文支持包) 由于安装英文版的系统不支持中文&#xff0c;出现中文文件名乱码。 下面操作…

Linux-centos安装MySQL8.0.22连接驱动文件mysql-connector-java-8.0.22-1.el7.noarch.rpm

目录 1、下载地址 2、选择版本 3、安装驱动 1、下载地址 https://downloads.mysql.com/archives/c-j/ 2、选择版本 选择对应的版本&#xff0c;这里系统选择Redhat系列Linux7&#xff0c;驱动版本选择8.0.22&#xff0c;点击download下载到本地再上传至服务器安装。 也可…

Mysql-MHA 安装过程中遇到的问题 :报错rpm -ivh mha4mysql- manager- 0.56-0.el6.noarch.rpm错误:依赖检测失败:

rpm -ivh mha4mysql- manager- 0.56-0.el6.noarch.rpm 错误&#xff1a;依赖检测失败&#xff1a; perl(Config::Tiny) 被 mha4mysql-manager-0.56-0.el6.noarch 需要 perl(Log::Dispatch) 被 mha4mysql-manager-0.56-0.el6.noarch 需要 perl(Log::Dispatch::File) 被 mha4mysq…

Linux下载并安装rabbitmq-server-3.6.5-1.noarch.rpm

目录 1.安装rabbitmq所需要的依赖包 2.下载安装包 3.安装服务命令 4.修改配置 5.启动rabbitmq 6.rabbitmq控制台安装 7.访问你的虚拟机 ip:15627 会出现下面的页面 用户名和密码都是 guest 8.常用命令 1.安装rabbitmq所需要的依赖包 yum install build-essential o…