JNA 中的unsigned 型別對映
阿新 • • 發佈:2019-01-07
做JNA的時候遇到了unsigned int如何對映的問題,因為java中沒有unsigned 型別,還在想要不要就用普通的int型,然後最後取abs一下
嘗試了一下,發現最後返回的值不對。
DLL中的程式碼如下:
unsigned int add(unsigned int first,unsigned int second) {
printf("(c) test jna : %u + %u = %u\n", first, second, first + second);
return first + second;
}
jna呼叫如下:
int a=0x80000000; int b=0x70000000; int c=(TestJnaLib.INSTANCE.add(a, b)); //c=c& 0xffffffffl; System.out.println("a+b="+c+" ");
結果如下所示:
a+b=-268435456
(c) test jna : 2147483648 + 1879048192 = 4026531840
改為c=Math.abs(c);
a+b=268435456
答案依然是錯誤的,想想也是,本來就是32位,一位用去做符號位了,直接取正數,會丟失原有的資料,因為符號位沒了。
上網一查之後,發現使用long來代替unsigned int,但是有問題時,long是64位,如果直接代替了必然會出錯。
經過測試,在JNA中add函式應該這樣宣告:
int add(int first, int second); long add(int first, int second);
兩種都可以,但是感覺第二中由於long是64位,應該會覆蓋掉棧中的資料,但是呼叫成功,很是不解,但是推薦使用第一種,這樣在DLL中執行程式碼沒有問題,當如果需要取值的時候,應該這樣取:
int a=0x80000000; int b=0x60000000; System.out.println(a+" "+b+" "); long c=(TestJnaLib.INSTANCE.add(b, a)); long d=a+b; d=d& 0xffffffffl; c=c& 0xffffffffl; System.out.println("a+b="+c+" "); System.out.println("d:"+d+" ");
一定要用long型別通過&操作符來把低32位取出來。通過對比d,兩者是相同的,說明這種方法是對的。
在結構體中如果出現了unsigned int型,一定用int來代替,而不是long,否則你會發現會覆蓋掉某些資料成員。