JNI使用。 “符号查找错误”

时间:2013-06-26 14:26:22

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

我尝试将md5共享库与Java Native Interface一起使用。 Here您可以查看来源。该库太大,无法与JNI一起使用。所以我想使用一个简单的附件库进行链接。在测试之后构建这个库后我得到了这个输出:

  java: symbol lookup error: ./libMd5bridge.so: undefined symbol: _Z3md5Ss

我尝试做LD_PRELOAD,但这不是结果。所以我在这里逐步发布我的所有行动。

1)我可以用非常简单的C ++代码调用我的md5库:

#include <iostream>
#include "md5.h"

int main(int argc, char* argv[]) {
    std::string passedValue;
    for(int i = 1; i < argc; i++)
     passedValue += argv[i];
    std::cout << "MD5 sum is: " << md5(passedValue) <<std::endl;
    return 0;
}

我将我的配件库命名为Md5bridge。因此,在下一个上下文中,我将这样称呼它。

2) Md5Bridge.java

public class Md5bridge{
     native public void MD5SUM(String[] passedValue);
}

3)获取Md5bridge头文件:

javac Md5Bridge.java && javah Md5Bridge

Md5bridge.h

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

#ifndef _Included_Md5bridge
#define _Included_Md5bridge
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     Md5bridge
 * Method:    MD5SUM
 * Signature: ([Ljava/lang/String;)V
 */
JNIEXPORT void JNICALL Java_Md5bridge_MD5SUM
  (JNIEnv *, jobject, jobjectArray);

#ifdef __cplusplus
}
#endif
#endif

4)准备从第一步到使用jni的代码:

#include <iostream>
#include "md5.h"
#include "Md5bridge.h"
JNIEXPORT void JNICALL Java_Md5bridge_MD5SUM
  (JNIEnv *, jobject, jobjectArray) {

    std::string passedValue;
    md5(passedValue);
}

5)编译为共享库:

g++ -shared -fPIC -I /opt/jdk1.6.0_45/include/ -I /opt/jdk1.6.0_45/include/linux -o libMd5bridge.so Md5bridge.cpp

6)准备一个简单的java程序来检查这个库:

   public class Main{
    static{
    System.loadLibrary("Md5bridge");
    }
    public static void main(String[] passedValue){
    new Md5bridge().MD5SUM(passedValue);
    }
    }

这个程序是用javac编译的,并且运行错误(我在这篇文章的头部写了一篇):

java: symbol lookup error: ./libMd5bridge.so: undefined symbol: _Z3md5Ss

我不熟悉java,我相信我在其中一个步骤中做错了。有什么想法吗?

更新: 添加extern“C”{} md5头文件后(感谢Chris Stratton)我有这个输出:

java: symbol lookup error: /home/andrey/Desktop/md5LIB/libMd5bridge.so: undefined symbol: md5

它看起来像错过了libmd5.so的链接。

$ ldd libMd5bridge.so 
    linux-vdso.so.1 =>  (0x00007fff38666000)
    libstdc++.so.6 => /usr/lib/libstdc++.so.6 (0x00007f2dfde3a000)
    libm.so.6 => /lib/libm.so.6 (0x00007f2dfdbb7000)
    libgcc_s.so.1 => /lib/libgcc_s.so.1 (0x00007f2dfd99f000)
    libc.so.6 => /lib/libc.so.6 (0x00007f2dfd619000)
    /lib64/ld-linux-x86-64.so.2 (0x00007f2dfe36b000)

如何链接两个库?

2 个答案:

答案 0 :(得分:1)

您的C实现必须如下:

#include <iostream>
#include "md5.h"
#include "Md5bridge.h"
#ifdef __cplusplus
extern "C" {//pay attention to this!
#endif 
JNIEXPORT void JNICALL Java_Md5bridge_MD5SUM
  (JNIEnv *, jobject, jobjectArray) {

   std::string passedValue;
   md5(passedValue);
}
#ifdef __cplusplus
} //pay attention to this
#endif 

但更好!使用已在JDK中实现的origin md5

以下是我的java代码的快照:

public class MD5Converter {

    private final static String ALGORITHM = "MD5";
    private final static String ENCODING = "UTF-8";

    public static String encryptToMd5(String normal) {
        MessageDigest digest;
        try {
            digest = java.security.MessageDigest.getInstance(ALGORITHM);
            digest.reset();
            digest.update(normal.getBytes(ENCODING));
            byte[] hash = digest.digest();
            return byteToHex(hash); //This is on you res to implement this 
        } catch (NoSuchAlgorithmException e) {
            final String err = "No such digest algorithm \"" + ALGORITHM + "\"";
            log.fatal(err, e);
            throw new IllegalStateException(err, e);
        } catch (UnsupportedEncodingException e) {
            final String err = "Unsupported encoding \"" + ENCODING + "\"";
            log.fatal(err, e);
            throw new IllegalStateException(err, e);
            }
        }
}

答案 1 :(得分:0)

您可以通过使用-l选项进行编译,将.so链接到其他库。如果库未安装在系统目录中,则可以使用-L指定目录。

例如,如果md5函数在工作目录的libmd5.so中,则将-L. -lmd5添加到编译器命令行:

  

g ++ -shared -fPIC -I /opt/jdk1.6.0_45/include/ -I /opt/jdk1.6.0_45/include/linux -o libMd5bridge.so Md5bridge.cpp -L。 -lmd5

由于这种情况下的库似乎非常小,因此将代码直接编译到“桥”库中而不是从另一个库加载它可能是有意义的。要执行此操作,只需将源文件添加到命令行:

  

g ++ -shared -fPIC -I /opt/jdk1.6.0_45/include/ -I /opt/jdk1.6.0_45/include/linux -o libMd5bridge.so Md5bridge.cpp MD5.cpp Whatever.cpp