3

我试图将 md5 共享库与 Java Native Interface 一起使用。在这里您可以查看来源。这个库太大,无法准备与 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)

如何链接两个库?

4

2 回答 2

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 

但更好!使用已经在 J​​DK 中实现的 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);
            }
        }
}
于 2013-06-26T14:36:39.003 回答
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 不管什么.cpp

于 2013-06-27T13:13:57.943 回答