java - 在单独的类中使用 Java native 方法时出现 UnsatisfiedLinkError

标签 java java-native-interface

我正在使用 Java native 接口(interface)来访问我用 C++ 创建的 DLL 中的多个 native 方法。我可以毫无问题地加载库,但只能调用位于 Java 应用程序的主类中的方法。换句话说,只要我声明方法并将 loadLibrary 命令放在主类中,一切都会完美运行。

但是,如果我尝试放置 loadLibrary 命令,并在单独的类中声明 native 方法,则会收到以下错误:

Exception in thread "main" java.lang.UnsatisfiedLinkError: sdr_api.SDR_API_Interface.SetDebugFlag_Native(Z)V
at sdr_api.SDR_API_Interface.SetDebugFlag_Native(Native Method)
at sdr_api.SDR_API_Interface.SetDebugFlag(SDR_API_Interface.java:74)
at sdr_api.SDR_API.main(SDR_API.java:183)

我尝试查看 JNI 文档来检查是否要求所有 native 方法都必须在主类中声明,但我找不到任何内容。我想将这些方法移到一个单独的类中,因为我需要在单独的线程中使用它们(我只需要该类的一个实例,因此如果需要,可以将它们声明为静态)。

我最好的猜测是,在特定类中声明方法会以某种方式改变它们在 DLL 中调用的方式,但我不知道如何改变。

任何帮助将不胜感激!

已编辑:添加了相关代码示例

SDR_API 类:

package sdr_api;

public class SDR_API 
{       
    public static void main(String[] args) 
    {

        try
        {
            double Rate = 2e5;
            double TxRxFreq = 5e9;
            double RxGain = 0;
            double TxGain = 0;

            SDR_API_Interface mySDRAPI = new SDR_API_Interface();
            mySDRAPI.SetDebugFlag(true);                              

            mySDRAPI.Setup_BS_Config("addr=192.168.10.2",Rate,Rate,TxRxFreq,TxRxFreq,RxGain,TxGain);
            mySDRAPI.Setup_MT_Config("addr=192.168.20.2",Rate,Rate,TxRxFreq,TxRxFreq,RxGain,TxGain);

         }
         catch(Exception e)
         {
        System.err.println("Exception thrown in main program: "+e);
         }
    }
}

SDR_API_Interface 类:

package sdr_api;

public class SDR_API_Interface 
{
    private native boolean Setup_MT_Config_Native(String _IP_Address, 
                                                  double _Rx_Rate, 
                                                  double _Tx_Rate,
                                                  double _Rx_Freq, 
                                                  double _Tx_Freq, 
                                                  double _Rx_Gain, 
                                                  double _Tx_Gain);
    private native boolean Setup_BS_Config_Native(String _IP_Address, 
                                                  double _Rx_Rate, 
                                                  double _Tx_Rate,
                                                  double _Rx_Freq, 
                                                  double _Tx_Freq, 
                                                  double _Rx_Gain, 
                                                  double _Tx_Gain);

    private native void SetDebugFlag_Native(boolean _DebugFlag);

    private native double[] TransmitUL_Native(double[] _TxWaveform);
    private native double[] TransmitDL_Native(double[] _TxWaveform);

    private boolean Debug_Flag = false;

    static 
    {
        System.loadLibrary("SDR_API");
    }

    public boolean Setup_MT_Config(String _IP_Address, 
                                    double _Rx_Rate, 
                                    double _Tx_Rate,
                                    double _Rx_Freq, 
                                    double _Tx_Freq, 
                                    double _Rx_Gain, 
                                    double _Tx_Gain)
    {
        return Setup_MT_Config_Native(_IP_Address, 
                                      _Rx_Rate, 
                                      _Tx_Rate,
                                      _Rx_Freq, 
                                      _Tx_Freq, 
                                      _Rx_Gain, 
                                      _Tx_Gain);        
    }

    public boolean Setup_BS_Config(String _IP_Address, 
                                    double _Rx_Rate, 
                                    double _Tx_Rate,
                                    double _Rx_Freq, 
                                    double _Tx_Freq, 
                                    double _Rx_Gain, 
                                    double _Tx_Gain)
    {
        return Setup_BS_Config_Native(_IP_Address, 
                                      _Rx_Rate, 
                                      _Tx_Rate,
                                      _Rx_Freq, 
                                      _Tx_Freq, 
                                      _Rx_Gain, 
                                      _Tx_Gain);         
    }

    public void SetDebugFlag(boolean _DebugFlag)
    {
        Debug_Flag = _DebugFlag;
        SetDebugFlag_Native(Debug_Flag);   
    }

    public boolean GetDebugFlag()
    {
        return Debug_Flag;
    }
}

JNI 机器生成的头文件:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class sdr_api_SDR_API */

#ifndef _Included_sdr_api_SDR_API
#define _Included_sdr_api_SDR_API
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     sdr_api_SDR_API
 * Method:    Setup_MT_Config_Native
 * Signature: (Ljava/lang/String;DDDDDD)Z
 */
JNIEXPORT jboolean JNICALL Java_sdr_1api_SDR_1API_Setup_1MT_1Config_1Native
  (JNIEnv *, jobject, jstring, jdouble, jdouble, jdouble, jdouble, jdouble, jdouble);

/*
 * Class:     sdr_api_SDR_API
 * Method:    Setup_BS_Config_Native
 * Signature: (Ljava/lang/String;DDDDDD)Z
 */
JNIEXPORT jboolean JNICALL Java_sdr_1api_SDR_1API_Setup_1BS_1Config_1Native
  (JNIEnv *, jobject, jstring, jdouble, jdouble, jdouble, jdouble, jdouble, jdouble);

/*
 * Class:     sdr_api_SDR_API
 * Method:    SetDebugFlag_Native
 * Signature: (Z)V
 */
JNIEXPORT void JNICALL Java_sdr_1api_SDR_1API_SetDebugFlag_1Native
  (JNIEnv *, jobject, jboolean);

/*
 * Class:     sdr_api_SDR_API
 * Method:    TransmitUL_Native
 * Signature: ([D)[D
 */
JNIEXPORT jdoubleArray JNICALL Java_sdr_1api_SDR_1API_TransmitUL_1Native
  (JNIEnv *, jobject, jdoubleArray);

/*
 * Class:     sdr_api_SDR_API
 * Method:    TransmitDL_Native
 * Signature: ([D)[D
 */
JNIEXPORT jdoubleArray JNICALL Java_sdr_1api_SDR_1API_TransmitDL_1Native
  (JNIEnv *, jobject, jdoubleArray);

#ifdef __cplusplus
}
#endif
#endif

最佳答案

@瑞安 仔分割析头文件可以看到 native 方法的名称是 sdr_api_SDR_API_SetDebugFlag_Native

来自 SDR_API_Interface 的调用是 - sdr_api.SDR_API_Interface.SetDebugFlag_Native

可能您使用不同的类创建了头文件 即 SDR_API 位于 sdr_api 包下。

实际上应该使用 Interface 类创建 - sdr_api.SDR_API_Interface.SetDebugFlag_Native

您所需要做的就是编译类 SDR_API_Interface,创建头文件,然后创建新的 dll 来使用它。

您正在使用的 dll 仍在引用 SDR_API 类中对 native 方法的调用。 它应该是SDR_API_Interface类。

如果您仍有疑问,请告诉我。

关于java - 在单独的类中使用 Java native 方法时出现 UnsatisfiedLinkError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7807340/

相关文章:

java - 从 JNI/NDK 将二维原始数组从 C 返回到 Java

java - 未找到 android 项目中的 native 库(UnsatisfiedLinkError)

java - Swing 的骨架应用程序

java - 如何找到 LocalDate java 之间的天数?

java - Hibernate 中 setCalendar 和 setCalendarDate 有什么区别?

android - 我们如何将动画/简单图像添加为视频的顶层并将其导出为 Android 中的单个视频?

java - JAXB。从字符串中获取 boolean 值

java - 在 WebView 中为我的 Android 应用程序使用自定义 (unicode) 字体

java - 使用 JNI 从 Java 到 native (C++) 代码使用 byte[]

java - 如何防止 JNI 覆盖 GetLastError()