我有一个自己的构建Eclipse插件,我需要调用一个C ++ DLL。
我尝试分两步完成: 1.通过调用C ++ dll的Java主程序在Eclipse-plugin之外 2.尝试将其插入我的插件(这就是问题所在)
主要Java代码HelloWorld.java。
class HelloWorld {
//public native void print(); //native method
public native String print(String msg); //native method
static //static initializer code
{
System.loadLibrary("CLibHelloWorld");
}
public static void main(String[] args)
{
//HelloWorld hw = new HelloWorld();
//hw.print();
String result = new HelloWorld().print("Hello from Java");
System.out.println("In Java, the returned string is: " + result);
}
}
通过命令编译: “C:\ Program Files \ Java \ jdk1.6.0_34 \ bin \ javac”HelloWorld.java
然后我通过以下方式为C ++ dll创建了一个h文件HelloWorld.h:
“C:\ Program Files \ Java \ jdk1.6.0_34 \ bin \ javah”HelloWorld
h文件如下所示:
#include <jni.h>
/* Header for class HelloWorld */
#ifndef _Included_HelloWorld
#define _Included_HelloWorld
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: HelloWorld
* Method: print
* Signature: (Ljava/lang/String;)Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_HelloWorld_print
(JNIEnv *, jobject, jstring);
#ifdef __cplusplus
}
#endif
#endif
现在是C ++ dll CLibHelloWorld.cpp:
#include "HelloWorld.h"
#include "jni.h"
#include "stdafx.h"
#include "tchar.h"
#import "..\ManagedVBDLL\bin\Debug\ManagedVBDLL.tlb" raw_interfaces_only
using namespace ManagedVBDLL;
JNIEXPORT jstring JNICALL Java_HelloWorld_print(JNIEnv *env, jobject thisObj, jstring inJNIStr) {
jboolean blnIsCopy;
const char *inCStr;
char outCStr [128] = "string from C++";
inCStr = env->GetStringUTFChars(inJNIStr, &blnIsCopy);
if (NULL == inCStr) return NULL;
printf("In C, the received string is: %s\n", inCStr);
env->ReleaseStringUTFChars(inJNIStr, inCStr);
return env->NewStringUTF(outCStr);
}
构建dll
当我运行java主程序时......一切正常!
我创建了一个应该调用C ++ dll的类:
package org.eclipse.ui.examples.recipeeditor.support;
import org.eclipse.jface.dialogs.MessageDialog;
public class HelloWorld {
public native String print(String msg); //native method
static //static initializer code
{
try {
System.loadLibrary("CLibHelloWorld"); //$NON-NLS-1$
} catch (Exception e) {
e.printStackTrace();
MessageDialog.openInformation(null, "HelloWorld", "HelloWorld Catch: " + e.getMessage());
}
}
}
并将其称为:
HelloWorld hw = new HelloWorld();
result = hw.print("Hi from Eclipse");
然后我在hw.print上得到了这个错误(完成了dll的加载):
java.lang.UnsatisfiedLinkError:org.eclipse.ui.examples.recipeeditor.support.HelloWorld.print(Ljava / lang / String;)Ljava / lang / String;
很长一段时间,但我该如何解决呢?
感谢。
答案 0 :(得分:0)
System.loadLibrary仅在LD_LIBRARY_PATH(Linux)或PATH(Windows)中可用时才加载库。您还需要尊重正确的名称。在Windows中不确定,但在linux中如果你像你一样加载CLibHelloWorld
,你的DLL应该被称为libCLibHelloWorld.so
。我猜有一个System.getNativeMethodName或类似的东西,所以你可以找到它。
无论如何,这不是我加载DLL的优先方式,因为你依赖于很多环境设置。相反,您可以使用System.load (dll_full_path)
加载DLL。它具有相同的效果,但你有更多的控制权。
但是,如果您使用此方法成功加载DLL,并在尝试调用本机方法时不断收到上述错误,请查看DLL依赖项。您应首先加载依赖项,然后加载要调用的lib。
例如,如果要加载dll1,这取决于dll2,这取决于dll3,你应该这样做:
System.load(dll3_path);
System.load(dll2_path);
System.load(dll1_path);
答案 1 :(得分:0)
本机代码中的方法名称必须与java-class的类和包名称相对应。 由于HelloWorld类已从default-package更改为org.eclipse.ui.examples.recipeeditor.support,因此您必须更改方法名称。
只需用新类重新运行javah即可获得正确的头文件。
顺便说一句,如果你使用像
这样的东西在OSGI-bundle中正确定义了dll,就不需要设置一个单独的库路径Bundle-NativeCode: mydll.dll ; osname=win32 ; processor=x86
并将dll包含在插件的根目录中。