Eclipse插件,调用C ++ DLL

时间:2012-11-10 12:13:31

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

我有一个自己的构建Eclipse插件,我需要调用一个C ++ DLL。

我尝试分两步完成: 1.通过调用C ++ dll的Java主程序在Eclipse-plugin之外 2.尝试将其插入我的插件(这就是问题所在)

  1. 在Eclipse插件之外。
  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主程序时......一切正常!

    1. 尝试将其插入我的Eclipse插件(这就是问题所在)
    2. 我创建了一个应该调用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;

      很长一段时间,但我该如何解决呢?

      感谢。

2 个答案:

答案 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包含在插件的根目录中。