另一个JNI,C ++,DLL,UnsatisfiedLinkError <native method =“”> </native>

时间:2011-12-07 16:00:51

标签: java c++ java-native-interface native unsatisfiedlinkerror

我一直在寻找2天,没有任何解决方案可以帮助我,所以我们再来一次:

如何解决JNI中的UnsatisfiedLinkError ...

所以这是我的java代码:

package org.lingenio.util;

import java.util.*;

public class PTAPIWrapperForOmegaT {

    private native String translateWithPTAPI(String sentence);

    private native void test();

    public PTAPIWrapperForOmegaT(String sentence) throws Exception{
        System.out.println(sentence);
        test();     
    }

    static {
        System.load("C:/Users/michael/Desktop/OmegaT/OmegaT2.3_src/native/PTAPIWrapperForOmegaT.dll");
    }
}

这是我的C ++代码:

    #include <iostream>
    #include <windows.h>
    #include <jni.h>
    #include "PTAPIWrapperForOmegaT.h"

    using namespace std;

    JNIEXPORT jstring JNICALL Java_PTAPIWrapperForOmegaT_translateWithPTAPI(JNIEnv *env, jobject obj, jstring sentence)
    {
/* stuff */
    }

    JNIEXPORT void JNICALL Java_PTAPIWrapperForOmegaT_test(JNIEnv *, jobject)
    {
        cout << "This comes from PTAPIWrapperForOmegaT.cpp test();" << endl;
    }


    int main(){
        return 0;
    }

头文件:

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

#ifndef _Included_PTAPIWrapperForOmegaT
#define _Included_PTAPIWrapperForOmegaT
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     PTAPIWrapperForOmegaT
 * Method:    translateWithPTAPI
 * Signature: (Ljava/lang/String;)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_PTAPIWrapperForOmegaT_translateWithPTAPI
  (JNIEnv *, jobject, jstring);

/*
 * Class:     PTAPIWrapperForOmegaT
 * Method:    test
 * Signature: ()V
 */
JNIEXPORT void JNICALL Java_PTAPIWrapperForOmegaT_test
  (JNIEnv *, jobject);

#ifdef __cplusplus
}
#endif
#endif

以及我如何构建它:

call g++ -Wl,--add-stdcall-alias -c -DBUILDING_EXAMPLE_DLL -I G:/Software/Java/jdk1.7.0_01/include -I G:/Software/Java/jdk1.7.0_01/include/win32 PTAPIWrapperForOmegaT.cpp
call g++ -shared -Wl,-kill-at -o PTAPIWrapperForOmegaT.dll -I G:/Software/Java/jdk1.7.0_01/include -I G:/Software/Java/jdk1.7.0_01/include/win32 PTAPIWrapperForOmegaT.cpp

最后,错误:

10211: Error: Uncatched exception in thread [Thread-14] 
10211: Error: java.lang.UnsatisfiedLinkError: org.lingenio.util.PTAPIWrapperForOmegaT.test()V 
10211: Error:   at org.lingenio.util.PTAPIWrapperForOmegaT.test(Native Method) 
10211: Error:   at org.lingenio.util.PTAPIWrapperForOmegaT.<init>(PTAPIWrapperForOmegaT.java:13) 
10211: Error:   at org.omegat.core.machinetranslators.LingenioTranslate.translate(LingenioTranslate.java:32) 
10211: Error:   at org.omegat.core.machinetranslators.BaseTranslate.getTranslation(BaseTranslate.java:64) 
10211: Error:   at org.omegat.gui.exttrans.MachineTranslateTextArea$FindThread.search(MachineTranslateTextArea.java:122) 
10211: Error:   at org.omegat.gui.exttrans.MachineTranslateTextArea$FindThread.search(MachineTranslateTextArea.java:102) 
10211: Error:   at org.omegat.gui.common.EntryInfoSearchThread.run(EntryInfoSearchThread.java:85) 

我不确切地知道这两行g ++,我认为第二行就足够了,但是一些教程也必须提供另一行,我保留了它。

我在Windows 7上,使用MingW和最新的Java(我相信1.7xxx)。

感谢任何帮助,我怀疑错误在于编译,但我不知道如何从这里开始。

修改

使用DependencyWalker查看dll我可以看到函数的名称就像我在.cpp文件中命名它们一样。当然,我从Java Wrapper中调用它们各自的名称,即test()。这可能是个问题吗?经常使用JNI的人可以告诉我这是否是正确的方法?

1 个答案:

答案 0 :(得分:4)

原来所有代码都没问题。实际上我做了错误编译头文件。您可以查看是否查看头文件的函数名称,即:

JNIEXPORT jstring JNICALL Java_PTAPIWrapperForOmegaT_translateWithPTAPI
  (JNIEnv *, jobject, jstring);

现在,看看你的Java文件的包成员资格,在我的情况下:

package org.lingenio.util;

因为我确实以错误的方式编译了头文件,JNI后来无法找到它正在寻找的符号,因为它实际上是在寻找这个:

JNIEXPORT jstring JNICALL Java_org_lingenio_util_PTAPIWrapperForOmegaT_translateWithPTAPI(JNIEnv *env, jobject obj, jstring sentence)

所以,那里的人们好好运抱同样的问题。我显然不是最优秀的Java程序员,这就是为什么我不得不长期担心这个问题。我应该首先以正确的方式编译我的头文件。 检查你的包和类路径!