我尝试将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)
如何链接两个库?
答案 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