有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

JNI调用时发生java UnsatifiedLinkError

我已经发现了很多关于与JNI结合的未满足LinkError的东西和问题,但是没有一个能帮到我。 我试图用Java隐藏一个窗口,所以我需要3个WinAPI函数

我有以下Java源代码:

package hide.window;
public class HideWindow {

    public native static int hideWindow(char[] windowTitle);

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

    public static void main(String[] args) {
        System.out.println("loadLibrary was successful");
        System.out.println(hideWindow("Session Window - Windows Internet Explorer".toCharArray()));
    }
}

然后我就把我的藏身之所藏在窗户里。h由javah生成。 有一件事我必须改变,因为它没有找到jni。h文件位于库路径中,因此我将其放入我的项目中,并以以下方式将其包括在内:

#include "jni.h"

这会导致我的错误吗 整个文件:

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

#ifndef _Included_hide_window_HideWindow
#define _Included_hide_window_HideWindow
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     hide_window_HideWindow
 * Method:    hideWindow
 * Signature: ([C)I
 */
JNIEXPORT jint JNICALL Java_hide_window_HideWindow_hideWindow
  (JNIEnv *, jobject, jcharArray);

#ifdef __cplusplus
}
#endif
#endif

我编译到DLL的代码如下所示:

#include <windows.h>
#include "hide_window_HideWindow.h"
using namespace std;

JNIEXPORT jint JNICALL Java_hide_window_HideWindow_hideWindow(char windowTitle[20])
{
    HWND windowHandle = FindWindowA(NULL, windowTitle);
    if (windowHandle){
        if (IsWindowVisible(windowHandle)){
            ShowWindow(windowHandle, 0);
            return 0;
        }
        else{
            ShowWindow(windowHandle, 1);
            return 1;
        }
    }
    return -1;
}

来自eclipse的调用与通过cmd进行的以下调用之间没有区别(路径正确):

java -Djava.library.path=.. hide.window.HideWindow

结果输出如下所示:

loadLibrary was successful
Exception in thread "main" java.lang.UnsatisfiedLinkError: hide.window.HideWindow.hideWindow([C)I
  at hide.window.HideWindow.hideWindow(Native Method)
  at hide.window.HideWindow.main(HideWindow.java:12)

这向我表明Java程序能够加载DLL,但在函数调用时失败。 我使用Eclipse作为java和Codeblocks作为C++ IDE,我已经在MIWW构建选项中添加了以下内容。p>

-Wl,--kill-at

我希望这个问题不要太愚蠢,提前谢谢你, 汤姆


共 (1) 个答案

  1. # 1 楼答案

    看看原型:

    JNIEXPORT jint JNICALL Java_hide_window_HideWindow_hideWindow
        (JNIEnv *, jobject, jcharArray);
    

    以及您的实施:

    JNIEXPORT jint JNICALL Java_hide_window_HideWindow_hideWindow
        (char windowTitle[20])
    

    当然有一个UnsatisfiedLinkError,因为您没有实现正确的函数

    所以你需要一些类似的东西:

    JNIEXPORT jint JNICALL Java_hide_window_HideWindow_hideWindow
        (JNIEnv* env, jobject thiz, jcharArray windowTitle)
    

    要从数组中获取字符,请查看GetPrimitiveArrayCritialGetCharArrayRegion

    生成的头在我看来也是可疑的,因为您将Java方法定义为static,但生成的原型想要传递一个jobject。。。静态方法需要jclass(因为静态方法调用没有实例对象):

    JNIEXPORT jint JNICALL Java_hide_window_HideWindow_hideWindow
        (JNIEnv* env, jclass clazz, jcharArray windowTitle)
    

    另一件事:Java的charjchar)是一个unsigned int16,C/C++char是有符号的或无符号的int8。因此,最好传递一个Stringjstring),并使用GetStringUTFChars以UTF8字符串的形式获取内容