使用java的jna调用c++的so文件

19 篇文章 0 订阅

整体描述

最近项目需要调用so文件,c++的同事给了一个so文件,在java端使用jna调用,记录一下具体操作和遇到的坑…本文的配置方法都是在linux下的配置,因为最后程序也是在linux下运行。

前期准备

1. 上传so文件

将so文件上传到java的lib目录下,一般会在/etc/profile文件里配置java的lib目录

2. 修改配置文件

如果这个so文件可能还要依赖其他的so文件,这个就要问下写so文件的人,看看他们是在哪个路径读取的so文件,我的项目里修改的是/etc/ld.so.conf文件,将so中依赖的其他so文件路径加上,修改之后使用ldconfig命令更新配置文件(这一步可以先跳过,如果调用so失败时再看)

java端代码

1. 引入jni依赖

        <dependency>
            <groupId>net.java.dev.jna</groupId>
            <artifactId>jna-platform</artifactId>
        </dependency>
        <dependency>
            <groupId>net.java.dev.jna</groupId>
            <artifactId>jna</artifactId>
            <version>4.4.0</version>
        </dependency>

2.创建jna类

此处需要注意的是:so文件名称为libtest.so,在代码中写so文件名称时,写test就可以

package com.thcb.jna;

import com.sun.jna.Library;
import com.sun.jna.Native;

/**
 * @author thcb
 */

public interface JnaTest extends Library {

    /**
     * jna库,test为so文件名称
     */
    public static final Test jnaTest = (JnaTest) Native.loadLibrary("jnatest", JnaTest.class);

    /**
     * 调用so文件,jna_test为so文件中的方法
     *
     * @param username        登陆用户名
     * @param passwd          登陆密码
     * @return
     */
    public abstract int jna_test(String username, String passwd) throws IllegalArgumentException;
}

3.调用so文件中的方法

      try {
             int result = JnaTest.jnaTest.jna_test(account, pwd);
             log.info("result:{}", result);
      } catch (Throwable t) {
           log.error("Exception:{}", t.toString());
      }

try catch很有必要,可以catch一些so文件调用的异常,避免程序崩溃,但是so文件中的方法抛出的异常在java是没法catch的,这块只能让c++开发时自己捕捉异常。

排坑指南

按照如上步骤,就可以实现java调用c++的so文件了,如果一切顺利没有报错,那么恭喜你后面的内容就不用看了(能一次就过,谢天谢地)

问题1

Exception:java.lang.UnsatisfiedLinkError: Unable to load library ‘jnatest’: Native library (darwin/libjnatest.dylib) not found in resource path
错误说明:意思就是so文件没找到
解决办法:这时候就要看看上面前期准备2提到的问题,还有so文件路径,java的lib路径的问题,java的路径一般配置在/etc/profile文件中

问题2

java: symbol lookup error: /app/data/packet/lib/librelayvoice.so: undefined symbol:XXXXXX
错误说明:后面XXXXX是找不到的方法,该错误就是没有找到so文件中的XXXXXX符号,一般来说是c++的函数名等。这个错误直接把java虚拟机jvm干崩了…
解决办法:一般情况下,这个问题就是so文件的问题了,需要重新编译so文件,可以用指令查看下so文件中使用其他的so:

ldd 【so文件名】

看看so文件中包含了哪些,我的问题就是我用的so文件中引用了其他的so文件里面的方法,但是引用的这个so文件并没有链接到我用的so文件中,导致报错。我找c++的同事要了cpp源码和.h文件,然后自己编译一下:

g++  -shared  -fPIC -L./lib - test234.so -o libtest.so test.cpp

通过cpp生成so文件,test234.so就是我的so需要引用的其他so文件的名称,前面的 -L./lib,/lib就是test234.so的目录,后面的libtest.so是生成so文件的名称,test.cpp就是cpp源文件。

使用新编译的so文件替换之前的so文件,需要重新启动一下java程序,就可以了,说明就是这个问题。

JNAJava Native Access)是一种 Java 编程语言的框架,它允许 Java 应用程序直接调用本地动态库(例如 .so 文件)中的函数。 下面是一个简单的示例程序,演示如何使用 JNA 调用本地动态库中的函数: 1. 首先,需要在 Java 代码中定义一个接口,用于描述要调用的本地函数。例如,如果要调用名为 `my_function` 的函数,可以定义如下接口: ```java import com.sun.jna.Library; import com.sun.jna.Native; public interface MyLibrary extends Library { MyLibrary INSTANCE = (MyLibrary) Native.loadLibrary("my_library", MyLibrary.class); void my_function(); } ``` 2. 然后,在 Java 代码中调用这个接口中定义的函数即可: ```java MyLibrary.INSTANCE.my_function(); ``` 3. 最后,需要将本地动态库文件(例如 my_library.so)放置到 Java 应用程序能够访问的路径中,例如运行程序时加上 `-Djava.library.path=/path/to/lib` 参数,或者将库文件放置到系统默认的库路径中。 需要注意的是,JNA调用本地函数时需要遵循特定的命名约定,例如在 Linux 系统下默认使用 C 语言的命名约定。如果本地函数使用了不同的命名约定,需要在 Java 代码中显式地指定。例如,如果本地函数使用C++ 语言的命名约定,可以在接口中添加 `@Name` 注解,例如: ```java public interface MyLibrary extends Library { MyLibrary INSTANCE = (MyLibrary) Native.loadLibrary("my_library", MyLibrary.class); @Name("my_function") void myFunction(); } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值