最近一个 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.<init>(JNative.java:512)
-
at org.xvolks.jnative.JNative.<init>(JNative.java:440)
-
at JNAtest.testDll(JNAtest.java:18)
-
at JNAtest.main(JNAtest.java:47)
于是 JNative 在 64 位系统上基本上没有办法,于是转向 JNA。
JNA 的好处在于,代码都是基于 Java,对于 Java 程序员来说简单易懂。
在开发过程中,目前遇到了一些问题,总结就是 DLL 返回 char 时,Java 端解析出现乱码。(甚至是英文字符串都乱码)
首先是 DLL 内的两个方法:
-
char test1(){
-
char buf[100] = "helloworld" ;
-
return buf ;
-
}
-
char test2(){
-
reutrn "helloworld" ;
-
}
Java 代码如下:
-
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
-
}
-
}
通过 Java 解析 test1() 方法时,每次都会出现乱码,即使返回的字符串是纯英文。但是 test2() 方法就没有出现问题。
此问题困扰了挺久的时间,后来查找 JNA API 和 上网查找,经过测试,解决了 test1() 方法解析乱码的问题。
这里需要修改 DLL 和 Java 代码。
DLL 代码修改后如下:
-
void test1(char buf){
-
char temp[100] = "helloworld" ;
-
memcpy(buf, temp, strlen(temp)) ;
-
return ;
-
}
Java 代码修改如下:
-
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");
-
}
-
}
通过 Java 获取 char 字符串,必须要通过 Java 传入一个 com.sun.jna.Pointer 指针变量,然后在 DLL 中将值赋给此指针变量,然后通过此指针变量获取值。
至此,一直困扰的乱码问题解决。
很多资料都说 DLL 返回 char 在 Java 中通过 String 便可以接受,但是目前测试,没有通过。