Linux下JAVA使用JNA调用C++的动态链接库(g++或者gcc编译的.so文件)

news2025/3/1 23:57:39

目录

前言        

一、准备工作

二、JAVA项目加载JNA

三、JNA的使用 

3.1 生成.so文件

        3.1.1 gcc生成的.so

         3.1.2 g++生成的.so

3.2 JNA调用.so       

四、JAVA与C++的类型对应

五、总结


前言        

        在没上班之前,我曾在CSDN写过《程序员的自我修养》的读书笔记,那真是一段难以忘记的时光,封控在学校里,也没法出去玩,无聊到只能看书打发时间,说多了。这本书是关于Linux的下程序的如何编译、链接的,真的推荐大家去看看,虽然我也没看完,读书笔记也才更到一半,真是遗憾,愿我能抽出时间把它啃完吧,我愿意用我的头发来换一点点知识,谁让我的头发很多掉不完呢,哈哈哈哈。

        转眼间到了工作,时间过的非常快,真是体会到了人生越过越快。这期间也装模作样的参加了公司的培训,接触到了很多非常优秀的人,觉得自己什么也不是,什么也没有,意难平。但其实每个人都有自己的命格,不属于你的永远不要强求,没必要给自己添堵,北方人的性格和思想大致就是这样吧。

        说了这么多,有什么关联呢?那就是我接到了工作后的首个Linux下的任务,曾经在学校搭建服务器、跑模型、给学弟学妹们建用户、维护等等都是在Linux下,本人对这个操作系统有较深的了解,而这次的任务是使用JAVA语言调用C++编译连接好的so动态链接库,Linux下的动态链接库就相当于Windows下的dll文件例如:msvcruntime.dll和Kernal32.dll,所以非常重要,是程序run起来后随时要加载的模块。

        工作的朋友可能会知道一个团队里有许许多多的角色,有产品、测试、开发、管理、HR等等,而开发可根据不同语言分为JAVA、C++、Python等等,他们之间也需要互相支撑,比如JAVA需要调用我们C++的方法,我的任务不就来了么。说实话,这方面的内容网上的资料真的很少,而且有错误、不统一、不全面、不明了,所以我决定更这一篇。

一、准备工作

        1. Linux操作系统,可以是Ubuntu、Centos、银河麒麟国产化系统等。可以在虚拟机上搞。

        2. JNA包,他在JNI上封了一层,可以使用封装好的方法调用so。

        JNA介绍:JNA提供工具用于调用c/c++动态函数库(如Window的dll以及linux的so)而不需要编写任何Native/JNI代码。开发人员只要在一个java接口中描述目标函数库的函数与结构,JNA将自动实现Java接口方法到函数的映射。

JNA github地址:

                        GitHub - java-native-access/jna: Java Native Access

嫌github慢可以在Gitcode镜像里下载:

                        mirrors / java-native-access / jna · GitCode

        3. JAVA的IDE,我是用的是JetBrains IntelliJ IDEA,下个Linux社区版就可以了。具体如何配置看下面链接(一定按里面要求来):

Java小白必会!Intellij IDEA安装、配置及使用详细教程_一一哥Sun的博客-CSDN博客_idea安装配置教程

        4. JAVA11,并安装到Linux系统上,配置好环境变量,具体可以看一下链接:

推荐:LINUX 安装 JAVA11_dubhe_zhao的博客-CSDN博客_linux安装java11

二、JAVA项目加载JNA

        我默认你已经搭载好了在Linux下的JAVA环境。那么开始加载JNA吧。

        我们在github或者gitcode上下载好的JNA如下图,为jna-5.12.1.jar,你放然可以下载其他版本,不过我没试过较低的版本是否能行。

        接下来创建一个JAVA项目,可以参考上面的链接如何创建JAVA项目,这个就不贴了,太麻烦了。然后,选择file下的Project Structure,如下:

        

        接着按照步骤选择Modules->Dependences->JAR or Directories

         

        接着选择自己下载好的JNA:

        接着点击加载的jna,点击ok即可。

         

        我们可以在External Libraries中看到加载好的.jar格式的JNA包,点开里面有两个文件夹:com.sun.jna和META-INF,我们要import的方法全在com.sun.jna中,我们只关注这个就好了。

        加载成功,庆祝一下,真不容易。

三、JNA的使用 

3.1 生成.so文件

        3.1.1 gcc生成的.so

        在JNA使用前我们需要制造一个.so文件,先随便写一个gcc可以编译过的代码,gcc不链接的话仅仅可以处理简单的C语言代码,如果想使用gcc编译C++代码,需要加extern “C”字段或者链接到需要使用的库,这个就很麻烦了,但我还是说说吧,毕竟自己卡在这块很久,需要给后来者清除这坑爹的障碍。

#include <stdio.h>
#include <stdlib.h>

int add(int a,int b);
int add3(int a,int b,int c);


int add(int a,int b) {
 
  return a+b;
}

int add3(int a,int b,int c)
{
  return a+b+c;
}

        接着分别运行:

        

gcc -fpic -c Hello.c

gcc -shared -o libHello.so Hello.o

        这样就完成了gcc生成so文件,我们的文件为libHello.so。 

      

         3.1.2 g++生成的.so

        g++这里我们重新写一份cpp代码,新建一个文件夹分别建立test.h和test.cpp:

        test.h:

#ifndef TEST_H_
#define TEST_H_
int add(int a, int b);
#endif

       test.cpp:

#include <stdio.h>
int add(int a, int b)
{
	printf("test_func ==> a = %d, b = %d\n", a, b);
	return (a+b);
}

       

        使用如下命令来生成g++的so:         

g++ test.cpp -fPIC -shared -o libHello.so

3.2 JNA调用.so       

        到这里我们使用JNA库里的两个文件即可,Native和Library。Native主要负责.so文件的加载,我们可以调用Native.load()来加载生成好的so文件。Library负责声明so中的方法供后续调用。下面我们使用一个具体的代码片段来说明。

        以下调用gcc生成的so文件:

import com.sun.jna.Library;
import com.sun.jna.Native;
/**
 * 一个java类
 * 运行环境是linux,需要打包生成jar文件放到linux环境运行
 */
public class HelloJNA {

    /**
     * 定义一个接口,默认的是继承Library ,如果动态链接库里的函数是以stdcall方式输出的,那么就继承StdCallLibrary
     * 这个接口对应一个动态链接(SO)文件
     */
    public interface LibraryAdd extends Library {
        LibraryAdd LIBRARY_ADD = Native.load("/home/bowen/Desktop/testhello/libHello.so", LibraryAdd.class);

        /**
         * 接口中只需要定义你要用到的函数或者公共变量,不需要的可以不定义
         * 映射libadd.so里面的函数,注意类型要匹配
         */
        int add(int a, int b);
    }

    public static void main(String[] args) {
        // 调用so映射的接口函数
        int add = LibraryAdd.LIBRARY_ADD.add(10, 15);
      
        System.out.println("a,b相加结果:" + add);

    }
}

        下面来解析这段代码,前面两个import引入JNA中的方法,如果成功加载JNA是没有问题的。接着声明了HelloJNA,与.class文件名同名。在HelloJNA类中首先声明了一个接口方法,这个接口继承Library,那就继承了JNA的Library里面的方法。这个接口主要任务为加载so文件并将内部需要使用的函数导出,不要忘了加载so后还要声明一下函数名,这个和C++中.h声明函数一致,后续需要这个函数名去查找其在so中的地址,通过地址偏移锁定到方法。在Main中我们可以调用这个接口,为这个接口传入指定参数,它会回传一个返回值,到这调用so成功。

        同样调用g++却报错,错误如下:

Exception in thread "main" java.lang.UnsatisfiedLinkError: Error looking up function 'add': /home/bowen/Desktop/testhello1/libHello.so: undefined symbol: add
	at com.sun.jna.Function.<init>(Function.java:252)
	at com.sun.jna.NativeLibrary.getFunction(NativeLibrary.java:604)
	at com.sun.jna.NativeLibrary.getFunction(NativeLibrary.java:580)
	at com.sun.jna.NativeLibrary.getFunction(NativeLibrary.java:566)
	at com.sun.jna.Library$Handler.invoke(Library.java:243)
	at com.sun.proxy.$Proxy0.add(Unknown Source)
	at HelloJNA.main(HelloJNA.java:29)

        这个问题可以这样描述。gcc和g++编译后的so文件不太一样,具体就是so内部的函数Symbol不一样。gcc编译过后so里的函数名或者方法名就是我们声明的那样,具体如下:

使用命令:nm -g libHello.so

获得:        

[bowen@localhost testhello]$ nm -g libHello.so 
0000000000000675 T add
0000000000000689 T add3
0000000000201028 B __bss_start
                 w __cxa_finalize@@GLIBC_2.2.5
0000000000201028 D _edata
0000000000201030 B _end
00000000000006a8 T _fini
                 w __gmon_start__
0000000000000540 T _init
                 w _ITM_deregisterTMCloneTable
                 w _ITM_registerTMCloneTable
                 w _Jv_RegisterClasses

        这个是64位Linux的so文件内部地址与符号的对应关系,想搞懂这个得看候捷老师的“程序的生前死后”,简直了。从这个结果我们能获得到函数add和add3,这个符号就是我们之前定义的函数名,那么我们在java中声明int add()当然没问题,可以调用起来。 

        但是我们看看g++生成的so内部长什么样:

        [bowen@localhost testhello1]$ nm -g libHello.so 
0000000000201030 B __bss_start
                 w __cxa_finalize@@GLIBC_2.2.5
0000000000201030 D _edata
0000000000201038 B _end
0000000000000708 T _fini
                 w __gmon_start__
0000000000000588 T _init
                 w _ITM_deregisterTMCloneTable
                 w _ITM_registerTMCloneTable
                 w _Jv_RegisterClasses
                 U printf@@GLIBC_2.2.5
00000000000006d5 T _Z3addii     

        它的add函数的符号变为_Z3addii,这个是g++为了防止符号重名给它生成的新名字,它的明明规则在《程序员的自我修养》里有说明,比如_Z、3、add、i、i这些都是根据原函数的特性而形成的。

        那么我们将代码中的add替换为_Z3addii可不可以呢?

a,b相加结果:25
test_func ==> a = 10, b = 15

        成功了,我们的猜测是正确的。

四、JAVA与C++的类型对应

        Java与C++类型对照表:

java类型c++类型原生表现
booleanint32位整数
bytechar8位整数
charwchar_t平台依赖
shortshort16位整数
intint32位整数
longlong long,__int6464位整数
floatfloat32位浮点数
doubledouble64位浮点数
Buffer/Pointerpointer平台依赖(32或64位指针)
<T>[](基本数据类型数组)pointer/array32位或64位指针(参数、返回值)邻接内存(结构体成员)
Stringchar*\0结束的数组(native encoding or jna.encoding)
WStringwchar_t*\0结束的数组(Unicode)
String[]char**\0结束的数组的数组
WString[]wchar_t**\0结束的宽字符数组的数组
Structurestruct*/struct指向结构体的指针(参数或返回值)(或者明确指定是结构体指针)、结构体(结构体的成员)(或明确指定是结构体)
Unionunion等同于结构体
Structure[]struct[]结构体的数组,邻接内存
Callback(*fp)()Java函数指针或原生函数指针
NativeLonglong平台依赖(32或64位整数)

        demo展示:
        待更新。。。

五、总结

        总的来说java使用JNA调用C++生成的so库没有问题,后续需要java调用更加复杂的C++代码,如类中的方法、vector等。java在调用so时主要使用symbol来定位方法,另外,我们在C++代码中加入extern “C”到底改变了什么也需要探究。祝大家周末愉快,疫情期间享受世界杯。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/38735.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Centos7下载安装Tomcat应用服务器

Centos7安装Apache-Tomcat-9 文章目录Centos7安装Apache-Tomcat-91. 前言2. 官网下载Apache-Tomcat3. 使用xftp工具或者rz工具传输文件4. 解压Tomcat文件并启动Tomcat文件5. 关闭防火墙并访问Apache-Tomcat6. 可以修改端口号7. 开放端口1. 前言 Apache-Tomcat服务器是一个免费的…

Android 开发人脸识别之自动识别验证码功能讲解及实现(超详细 附源码)

需要源码和图片集请点赞关注收藏后评论区留下QQ或者私信~~~ 一、自动识别验证码 验证码图片中最简单的是数字验证码&#xff0c;一张再普通不过的验证码拿到之后要进行以下步骤的处理 1&#xff1a;首先对图片适当裁剪&#xff0c;先去掉外部的空白区域&#xff0c;再把每个数…

Oracle日志复制—国产自研Beedup(基于日志结构化数据复制产品)

Beedup能够实现大量交易数据的实时捕捉、变换和投递&#xff0c;实现源数据库与目标数据库的数据同步&#xff0c;保持亚秒级的数据延迟。 一、Beedup产品概述 由北京灵蜂纵横软件有限公司研发的数据库实时复制软件Beedup&#xff0c;提供数据库Oracle异地容灾备份&#xff0…

Vue笔记_01双向数据绑定原理

[1]什么叫双向数据绑定&#xff1f; 视图中的数据发生了变化&#xff0c;data中的数据也要对应改变&#xff1b;data中的数据发生了变化&#xff0c;视图上的数据也要对应改变&#xff1b; [2]双向绑定原理 vue 双向数据绑定是通过 数据劫持 结合 发布订阅模式的方式来实现的…

Pip版本问题导致Python模块安装失败

文章目录起因解决方案前言输入 %APPDATA%创建 pip.ini终端执行命令安装包结语起因 今天在视频号平台看到有小姐姐直播讲爬虫技术&#xff0c;我一看这不是挺简单的吗&#xff1f;于是我就想到自己四年前的今天刚进大一的时候最开始学的爬虫技术&#xff0c;后来因为一些事情跑…

怎么证明前端数据加密的三种方式

导读&#xff1a;前端最常见的三大加密方式https&#xff0c;SSH和MD5&#xff0c;这篇文章带你走进三大加密方式的原理对比。 1.https 1.1原理 A.就是在http加入SSL层&#xff0c;是http安全的基础; B.htts协议是在http基础上加了SSL协议; C.使用443端口&#xff0c;http是80…

2022CTF培训(三)windowslinux安卓平台调试机制原理

附件下载链接 windows平台调试机制原理 手动编写一个简易调试器 创建待调试进程 使用 CreateProcess 函数创建待调试进程&#xff0c;创建时指定 dwCreationFlags 参数为 DEBUG_ONLY_THIS_PROCESS 将会告诉操作系统我们需要让当前调用者&#xff08;线程&#xff09;接管所…

(十)centos7案例实战——实现nginx代理访问redis服务

前言 本节内容是关于实现nginx代理访问redis服务&#xff0c;由于在实际生产开发环境中&#xff0c;我们并不想将我们的中间键服务暴露在公网环境中&#xff0c;或者只能在内网环境中使用&#xff0c;例如本节内容&#xff0c;我们将redis安装到本地环境&#xff0c;但是又有需…

链表中倒数第k个结点、反转链表、合并两个排序的链表、树的子结构、删除链表中重复的结点

文章目录1、链表中倒数第k个结点2、反转链表3、合并两个排序的链表4、树的子结构5、 二叉树的镜像6、删除链表中重复的结点1、链表中倒数第k个结点 本题考点&#xff1a; 链表&#xff0c;前后指针的使用&#xff0c;边界条件检测 牛客链接 题目描述&#xff1a; 输入一个链表…

JVM【八股文】

JVM【八股文】 JVM内存区域划分 程序计数器栈堆方法区 一块大的区域&#xff0c;需要根据功能&#xff0c;来划分不同的小区域。 JVM内存是从操作系统里申请来的&#xff0c;之后堆这部分区域进行了划分。 1.程序计数器 内存中最小的区域&#xff0c;保存了下一条要执行指令…

android-加壳加固

title: android-加壳加固 categories: Android tags: [android, 加壳, 加固, 混淆] date: 2022-06-20 18:00:23 comments: false mathjax: true toc: true android-加壳 前篇 Android之Apk加壳 - https://blog.csdn.net/LVXIANGAN/article/details/84956476Android动态加载Dex…

李沐论文精度系列之七:Two-Stream双流网络、I3D

文章目录一、双流网络1.1 前言1.2 网络结构1.3 光流(Optical flow)1.3.1 什么是光流1.3.2 如何利用光流1.3.3 双向光流&#xff08;Bi-directional optical flow&#xff09;1.3.4 光流的局限性及和对应的预处理&#xff08;抽取&#xff09;方式1.3.5 视频模型测试1.4 实验1.4…

✿✿✿JavaScript基本语法一

目 录 1.js的发展史&#xff08;闲聊版&#xff09; 2.浏览器分成两部分&#xff1a;渲染引擎和 JS 引擎 3.js与html的关系以及结合方式 (1)js与html的关系 (2)js与html结合方式 4.JavaScript注释 5.js中的基本数据类型 6.js中的变量 7.运算符&#xff08;自动类型转…

9.前端笔记-CSS-盒子模型-border和padding

页面布局的三大核心&#xff1a; 盒子模型浮动定位 1、盒子模型 1.1 盒子模型组成 盒子模型本质还是一个盒子&#xff0c;包括边框border、外边距margin、内边距padding和实际内容content 1.1.1 边框border 组成 组成&#xff1a;颜色border-color、边框宽度border-wid…

518. 零钱兑换 II【完全背包:求组合数】

518. 零钱兑换 II 给你一个整数数组 coins 表示不同面额的硬币&#xff0c;另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额&#xff0c;返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位…

C++11 右值,右值引用,移动构造,移动赋值

目录 一、左值&#xff0c;左值引用&#xff0c;右值&#xff0c;右值引用的相关概念&#xff1a; 1. 什么是左值&#xff0c;什么是左值引用&#xff1f; 2. 什么是右值&#xff0c;什么是右值引用&#xff1f; 3. 右值的属性是右值&#xff0c;右值引用的属性是左值 4. …

棒子老虎鸡-第12届蓝桥杯Scratch选拔赛真题精选

[导读]&#xff1a;超平老师计划推出Scratch蓝桥杯真题解析100讲&#xff0c;这是超平老师解读Scratch蓝桥真题系列的第86讲。 蓝桥杯选拔赛每一届都要举行4~5次&#xff0c;和省赛、国赛相比&#xff0c;题目要简单不少&#xff0c;再加上篇幅有限&#xff0c;因此我精挑细选…

研究生有限元仿真应用中存在的问题与对策

作者&#xff1a;尚晓江 导读&#xff1a;有限元分析软件作为计算工具&#xff0c;在科研和工程领域都有广泛应用&#xff0c;而多数用户是在研究生阶段开始接触和使用这些计算软件的。本文以ANSYS结构分析为例&#xff0c;对现阶段研究生应用有限元分析软件的现状和存在的问题…

无人机设计仿真--在Isight平台上进行的基于CST参数化+Xfoil的无人机翼型优化

作者&#xff1a;Graychen 一、工程背景 翼型的选型和设计是飞行器气动设计中的一项基础性工作&#xff0c;翼型对飞行器的气动性能具有根本性的影响。现在高性能飞行器已不再从翼型库中选择适用翼型后直接使用&#xff0c;而是以现有翼型作为基准翼型进行气动优化&#xff…

java基本语法 下

目录 运算符 运算符&#xff1a;算术运算符 运算符&#xff1a;赋值运算符 运算符&#xff1a;比较运算符 运算符&#xff1a;逻辑运算符 运算符&#xff1a;三元运算符 运算符的优先级 程序流程控制 概念 顺序结构 if-else结构 switch-case结构 循环结构 循环结构…