关闭

Java 通过 JNA 调用 DLL 返回 char * 字符串乱码问题的解决

标签: JNAJavachar乱码DLL
3286人阅读 评论(0) 收藏 举报
分类:

    最近一个 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 便可以接受,但是目前测试,没有通过。

0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:47118次
    • 积分:663
    • 等级:
    • 排名:千里之外
    • 原创:17篇
    • 转载:0篇
    • 译文:0篇
    • 评论:19条
    文章分类
    最新评论