hot*_*ure 2 java dll 64-bit java-native-interface unsatisfiedlinkerror
我正在使用JDK 1.7_25(64位)在64位Windows 8上运行Netbeans,遵循NetBeans的初始JNI的说明(https://netbeans.org/kb/docs/cnd/beginning-jni- linux.html)
这些说明适用于linux,但我认为Windows的原理是相同的(生成.dll文件而不是.so,在JDK中使用win32包含等)
我安装了Cygwin64以及Cygwin32.使用Cygwin64,我能够从我的C/C++动态库项目生成64位DLL.但是,当我调用System.load("path/to/JNITest.dll")时,我得到:
Exception in thread "main" java.lang.UnsatisfiedLinkError: C:\Users\Andrew\Documents\NetBeansProjects\JNITestLib\dist\JNITest.dll: %1 is not a valid Win32 application
at java.lang.ClassLoader$NativeLibrary.load(Native Method)
at java.lang.ClassLoader.loadLibrary1(ClassLoader.java:1957)
at java.lang.ClassLoader.loadLibrary0(ClassLoader.java:1882)
at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1843)
at java.lang.Runtime.load0(Runtime.java:795)
at java.lang.System.load(System.java:1061)
at jnitest.JNITest.main(JNITest.java:8)
Java Result: 1
Run Code Online (Sandbox Code Playgroud)
从我收集的内容来看,这通常是在32位虚拟机上加载64位应用程序时的情况,但我的netbeans.conf指向64位JVM.
另外,当我使用32位版本的Cygwin编译并运行时,我得到了
Can't load IA 32-bit .dll on a AMD 64-bit platform
Run Code Online (Sandbox Code Playgroud)
我很确定我正确地生成了DLL文件,它只是一个简单的HelloWorld printf来遵循JNI教程.我对JNI和C很新,所以我不确定从哪里开始调试.我所做的最好的是32位和64位DLL,我确保我的C编译器(Cygwin)是64位,我的JVM也是.
我很感激任何见解!
编辑:这是包含的文件
=== Java(JNITest.java)===
package jnitest;
public class JNITest {
public static void main(String[] args) {
System.out.println("JVM: " + System.getProperty("sun.arch.data.model"));
System.load("C:\\Users\\Andrew\\Documents\\NetBeansProjects\\JNITestLib\\dist\\JNITest.dll");
new JNITest().doHello();
}
public native void doHello();
}
Run Code Online (Sandbox Code Playgroud)
===生成的javah头文件(jnitest_JNITest.h)===
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class jnitest_JNITest */
#ifndef _Included_jnitest_JNITest
#define _Included_jnitest_JNITest
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: jnitest_JNITest
* Method: doHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_jnitest_JNITest_doHello
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
Run Code Online (Sandbox Code Playgroud)
=== C(JNITest.c)===
#include <jni.h>
#include "jnitest_JNITest.h"
JNIEXPORT void JNICALL Java_jnidemojava_Main_nativePrint
(JNIEnv *env, jobject obj)
{
printf("\nHello World from C\n");
}
Run Code Online (Sandbox Code Playgroud)
编辑:
问题似乎与DLL有关,因为我可以加载其他64位DLL就好了.我认为Cygwin可能是问题,因此我将编译器更改为MinGW-w64.它编译得很好,库加载,但现在我得到一个新的例外:
Exception in thread "main" java.lang.UnsatisfiedLinkError: jnitest.JNITest.doHello()V
at jnitest.JNITest.doHello(Native Method)
at jnitest.JNITest.main(JNITest.java:10)
Java Result: 1
Run Code Online (Sandbox Code Playgroud)
更多挖掘发现当ClassLoader在这里读取libs.size()时会抛出错误:
// Invoked in the VM class linking code.
static long findNative(ClassLoader loader, String name) {
Vector<NativeLibrary> libs =
loader != null ? loader.nativeLibraries : systemNativeLibraries;
synchronized (libs) {
int size = libs.size();
for (int i = 0; i < size; i++) {
NativeLibrary lib = libs.elementAt(i);
long entry = lib.find(name);
if (entry != 0)
return entry;
}
}
return 0;
}
Run Code Online (Sandbox Code Playgroud)
编辑:答案!
终于想通了.
首先,Cygwin64出了问题.使用不同的64位C编译器摆脱了有效的win32应用程序错误.
其次,我的JNITest.c文件的方法签名不正确.应该是:
Java_jnitest_JNITest_doHello
Run Code Online (Sandbox Code Playgroud)
代替
Java_jnitest_Main_doHello
Run Code Online (Sandbox Code Playgroud)
改变之后,它的工作原理!
(虽然我再也不能回答我自己的问题6个小时......所以dum de dum)
终于想通了.
首先,Cygwin64出了问题.使用不同的64位C编译器摆脱了有效的win32应用程序错误.
其次,我的JNITest.c文件的方法签名不正确.应该是:
Java_jnitest_JNITest_doHello
Run Code Online (Sandbox Code Playgroud)
代替
Java_jnitest_Main_doHello
Run Code Online (Sandbox Code Playgroud)
改变之后,它的工作原理!