我已经发现了很多关于 UnsatisfiedLinkError 与 JNI 结合的事情和问题,但它们都不能帮助我。 我正在尝试使用 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()));
}
}
然后我就得到了 javah 生成的 hide_window_HideWindow.h。 我必须更改一件事,因为它在库路径中找不到 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,并且我已经将以下内容添加到 mingw 构建选项中。
-Wl,--kill-at
我希望这个问题不是太愚蠢,提前谢谢你, 汤姆
最佳答案
看原型(prototype):
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)
要从数组中获取字符,请查看 GetPrimitiveArrayCritial
或 GetCharArrayRegion
。
生成的 header 对我来说也看起来很可疑,因为您将 Java 方法定义为 static
但生成的原型(prototype)想要传递 jobject
...静态方法需要jclass
(因为没有静态方法调用的实例对象):
JNIEXPORT jint JNICALL Java_hide_window_HideWindow_hideWindow
(JNIEnv* env, jclass clazz, jcharArray windowTitle)
另一件事:Java 的 char
(jchar
) 是一个 unsigned int16
,C/C++ char
是有符号的或无符号int8
。因此,最好传递 String
(jstring
) 并使用 GetStringUTFChars
以 UTF8 字符串形式获取内容。
关于java - JNI 调用时出现 UnsatisfiedLinkError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10347269/