最近一个 Java 项目需要调用公司的读卡器读取卡号。C 开发提供了一个读取卡号的 DLL。
Java 调用 DLL 无非三种方法:JNI、JNA、JNative
本来 C 开发测试时用了 JNative.jar 来调用 DLL,但是网路上的 JNative.jar 都是基于 32 位系统,其 jar 包中的两个 DLL 也均为 32 位。
在我本地尝试时都报错:
Exception in thread "main" java.lang.IllegalStateException: JNative library not loaded, sorry !
at org.xvolks.jnative.JNative.(JNative.java:512)
at org.xvolks.jnative.JNative.(JNative.java:440)
at JNAtest.testDll(JNAtest.java:18)
at JNAtest.main(JNAtest.java:47)
JNA 的好处在于,代码都是基于 Java,对于 Java 程序员来说简单易懂。
在开发过程中,目前遇到了一些问题,总结就是 DLL 返回 char * 时,Java 端解析出现乱码。(甚至是英文字符串都乱码)
首先是 DLL 内的两个方法:
char * test1(){
char buf[100] = "helloworld" ;
return buf ;
}
char * test2(){
reutrn "helloworld" ;
}
public class DLLUtil {
private static final String path = ConfigUtil.get("dllpath") ;
private static final String name = ConfigUtil.get("dllname") ;
public interface CLibrary extends Library{
//定义并初始化接口的静态变量
// path + File.separator + name = "F:/test/dll/test.dll"
CLibrary Instance=(CLibrary)Native.loadLibrary(path + File.separator + name,CLibrary.class);
String test1() ;
String test2() ;
}
public static void main(String[] args) {
System.setProperty("jna.encoding", "GBK");
String str1 = CLibrary.Instance.test1() ;
String str2 = CLibrary.Instance.test2() ;
System.out.println("test1_reply:" + str1) ; // 此处一直乱码
System.out.println("test2_reply:" + str2) ; // 此处正常为 helloworld
}
}
此问题困扰了挺久的时间,后来查找 JNA API 和 上网查找,经过测试,解决了 test1() 方法解析乱码的问题。
这里需要修改 DLL 和 Java 代码。
DLL 代码修改后如下:
void test1(char * buf){
char temp[100] = "helloworld" ;
memcpy(buf, temp, strlen(temp)) ;
return ;
}
public class DLLUtil {
private static final String path = ConfigUtil.get("dllpath") ;
private static final String name = ConfigUtil.get("dllname") ;
public interface CLibrary extends Library{
//定义并初始化接口的静态变量
// path + File.separator + name = "F:/test/dll/test.dll"
CLibrary Instance=(CLibrary)Native.loadLibrary(path + File.separator + name,CLibrary.class);
void test1(Pointer p) ;
}
public static void main(String[] args) {
System.setProperty("jna.encoding", "GBK");
// 首先定义指针,开辟内存空间,这里的内存空间根据返回的字符串来决定
Pointer p = new Memory(11) ;;
CLibrary.Instance.test1(p) ;
for(int i=0, sumi=11; i< sumi; i++){
System.out.print((char) p.getByteArray(0, 11)[i]);
}
System.out.println("\n");
}
}
至此,一直困扰的乱码问题解决。
很多资料都说 DLL 返回 char * 在 Java 中通过 String 便可以接受,但是目前测试,没有通过。