文章目录

  • 简介
  • String
  • Buffers,Memory,数组和Pointer
  • 可变参数
  • 总结

简介

JNA提供JAVA类型和native类型的映射关系,但是这一种映射关系只是一个大概的映射,我们在实际的应用中还有很多需要注意的事项,本文将会为大家详细讲解在使用类型映射中可能会出现的问题。一起来看看吧。

String

首先是String的映射,JAVA中的String实际上对应的是两种native类型:const char* 和 const wchar_t*。默认情况下String会被转换成为char* 。

char是ANSI类型的数据类型,而wchar_t是Unicode字符的数据类型,也叫做宽字符。

如果JAVA的unicode characters要转换成为char数组,那么需要进行一些编码操作,如果设置了jna.encoding,那么就会使用设置好的编码方式来进行编码。默认情况下编码方式是 “UTF8”.

如果是WString,那么Unicode values可以直接拷贝到WString中,而不需要进行任何编码。

先看一个简单的例子:

 char* returnStringArgument(char *arg) {return arg;
}wchar_t* returnWStringArgument(wchar_t *arg) {return arg;
}

上面的native代码可以映射为:

String returnStringArgument(String s);
WString returnWStringArgument(WString s);

再来看一个不同的例子,假如native方法的定义是这样的:

int getString(char* buffer, int bufsize);int getUnicodeString(wchar_t* buffer, int bufsize);

我们定义了两个方法,方法的参数分别是char* 和wchar_t*。

接下来看一下怎么在JAVA中定义方法的映射:

// Mapping A:
int getString(byte[] buf, int bufsize);
// Mapping B:
int getUnicodeString(char[] buf, int bufsize);

下面是具体的使用:

byte[] buf = new byte[256];
int len = getString(buf, buf.length);
String normalCString = Native.toString(buf);
String embeddedNULs = new String(buf, 0, len);

可能有同学会问了,既然JAVA中的String可以转换成为char*,为什么这里需要使用byte数组呢?

这是因为getString方法需要对传入的char数组中的内容进行修改,但是因为String是不可变的,所以这里是不能直接使用String的,我们需要使用byte数组。

接着我们使用Native.toString(byte[]) 将byte数组转换成为JAVA字符串。

再看一个返回值的情况:

// Example A: Returns a C string directly
const char* getString();
// Example B: Returns a wide character C string directly
const wchar_t* getString();

一般情况下,如果是native方法直接返回string,我们可以使用String进行映射:

// Mapping A
String getString();
// Mapping B
WString getString();

如果native code为String分配了内存空间,那么我们最好使用JNA中的Pointer作为返回值,这样我们可以在未来某些时候,释放所占用的空间,如下所示:

Pointer getString();

Buffers,Memory,数组和Pointer

什么时候需要用到Buffers和Memory呢?

一般情况下如果是基础数据的数组作为参数传到函数中的话,可以在JAVA中直接使用基础类的数组来替代。但是如果native方法在方法返回之后,还需要访问数组的话(保存了指向数组的指针),这种情况下使用基础类的数组就不太合适了,这种情况下,我们需要用到ByteBuffers或者Memory。

我们知道JAVA中的数组是带有长度的,但是对于native方法来说,返回的数组实际上是一个指向数组的指针,我们并不能知道返回数组的长度,所以如果native方法返回的是数组指针的话,JAVA代码中用数组来进行映射就是不合适的。这种情况下,需要用到Pointer.

Pointer表示的是一个指针,先看一下Pointer的例子,首先是native代码:

void* returnPointerArgument(void *arg) {return arg;
}void* returnPointerArrayElement(void* args[], int which) {return args[which];
}

接下来是JAVA的映射:

Pointer returnPointerArgument(Pointer p);Pointer returnPointerArrayElement(Pointer[] args, int which);

除了基本的Pointer之外,你还可以自定义带类型的Pointer,也就是PointerType. 只需要继承PointerType即可,如下所示:

public static class TestPointerType extends PointerType {public TestPointerType() { }public TestPointerType(Pointer p) { super(p); }}
TestPointerType returnPointerArrayElement(TestPointerType[] args, int which);

再看一下字符串数组:

char* returnStringArrayElement(char* args[], int which) {return args[which];
}wchar_t* returnWideStringArrayElement(wchar_t* args[], int which) {return args[which];
}

对应的JAVA映射如下:

String returnStringArrayElement(String[] args, int which);WString returnWideStringArrayElement(WString[] args, int which);

对应Buffer来说,JAVA NIO中提供了很多类型的buffer,比如ByteBuffer,ShortBuffer,IntBuffer,LongBuffer,FloatBuffer和DoubleBuffer等。这里以ByteBuffer为例,来看一下具体的使用.

首先看下native代码:

int32_t fillInt8Buffer(int8_t *buf, int len, char value) {int i;for (i=0;i < len;i++) {buf[i] = value;}return len;
}

这里将buff进行填充,很明显后续还需要使用到这个buffer,所以这里使用数组是不合适的,我们可以选择使用ByteBuffer:

int fillInt8Buffer(ByteBuffer buf, int len, byte value);

然后看下具体怎么使用:

TestLibrary lib = Native.load("testlib", TestLibrary.class);ByteBuffer buf  = ByteBuffer.allocate(1024).order(ByteOrder.nativeOrder());final byte MAGIC = (byte)0xED;lib.fillInt8Buffer(buf, 1024, MAGIC);for (int i=0;i < buf.capacity();i++) {assertEquals("Bad value at index " + i, MAGIC, buf.get(i));}

可变参数

对于native和JAVA本身来说,都是支持可变参数的,我们举个例子,在native方法中:

int32_t addVarArgs(const char *fmt, ...) {va_list ap;int32_t sum = 0;va_start(ap, fmt);while (*fmt) {switch (*fmt++) {case 'd':sum += va_arg(ap, int32_t);break;case 'l':sum += (int) va_arg(ap, int64_t);break;case 's': // short (promoted to 'int' when passed through '...') case 'c': // byte/char (promoted to 'int' when passed through '...')sum += (int) va_arg(ap, int);break;case 'f': // float (promoted to ‘double’ when passed through ‘...’)case 'g': // doublesum += (int) va_arg(ap, double);break;default:break;}}va_end(ap);return sum;
}

对应的JAVA方法映射如下:

public int addVarArgs(String fmt, Number... args);

相应的调用代码如下:

int arg1 = 1;
int arg2 = 2;
assertEquals("32-bit integer varargs not added correctly", arg1 + arg2,lib.addVarArgs("dd", arg1, arg2));

总结

本文介绍了在使用JNA方法映射中应该注意的一些细节和具体的使用问题。

本文的代码:https://github.com/ddean2009/learn-java-base-9-to-20.git

本文已收录于 http://www.flydean.com/05-jna-type-mapping-details-md/

最通俗的解读,最深刻的干货,最简洁的教程,众多你不知道的小技巧等你来发现!

欢迎关注我的公众号:「程序那些事」,懂技术,更懂你!

java高级用法之:JNA类型映射应该注意的问题相关推荐

  1. java高级用法之:JNA中的Structure

    文章目录 简介 native中的struct Structure 特殊类型的Structure 结构体数组作为参数 结构体数组作为返回值 结构体中的结构体 结构体中的数组 结构体中的可变字段 结构体中 ...

  2. java高级用法之:在JNA中使用类型映射

    文章目录 简介 类型映射的本质 TypeMapper NativeMapped 总结 简介 JNA中有很多种映射,library的映射,函数的映射还有函数参数和返回值的映射,libary和函数的映射比 ...

  3. java高级用法之:调用本地方法的利器JNA

    文章目录 简介 JNA初探 JNA加载native lib的流程 本地方法中的结构体参数 总结 简介 JAVA是可以调用本地方法的,官方提供的调用方式叫做JNI,全称叫做java native int ...

  4. java高级用法之:无所不能的java,本地方法调用实况

    文章目录 简介 JDK的本地方法 自定义native方法 总结 简介 相信每个程序员都有一个成为C++大师的梦想,毕竟C++程序员处于程序员鄙视链的顶端,他可以俯视任何其他语言的程序员. 但事实情况是 ...

  5. Java高级语法笔记-枚举类型

    用enum定义枚举类型,即定义一些常量. public enum Weekday{ SUNDAY,MONDAY,TUESDAY,WEDNESDAY, THURSDAY,FRIDAY,SATURDAY ...

  6. JAVA在IDEA中用JNA调用C++的dll动态链接库案例

    目录 引言 准备工作 1.JNA包 2.DLL文件 接口准备 接口参数说明 1.DLL路径 [报错]找不到文件 如何查看DLL文件的位数? 2.函数声明 如何查看DLL文件中有哪些函数? C++与JA ...

  7. java return用法_Java枚举的高级用法之多键值的映射使用

    枚举Enum单映射使用 做Java的各位仁兄姐妹都知道,Java通过HashMap,以及枚举提供了方便的K-V映射功能,例如 枚举单映射使用 但是如果遇到多个键值映射,例如K-K-V的形式怎么办呢?可 ...

  8. Java生成随机数的几种高级用法

    转载自 进阶 | Java生成随机数的几种高级用法! 言归正传,众所周知,随机数是任何一种编程语言最基本的特征之一.而生成随机数的基本方式也是相同的:产生一个0到1之间的随机数.看似简单,但有时我们也 ...

  9. java 不同类型 映射_如何使用Java泛型映射不同的值类型

    java 不同类型 映射 有时,一般的开发人员会遇到这样的情况,即他必须在特定容器内映射任意类型的值. 但是,Java集合API仅提供与容器相关的参数化. 例如,这将HashMap的类型安全使用限制为 ...

最新文章

  1. Centos7.2搭建Openstack的Swift组件,查看状态:No such file or directory
  2. 位运算一些简单的应用
  3. 【javascript】解决setTimeout不能传参的问题
  4. Xcode8打包上传后构建版本消失问题
  5. python混沌时间序列分析_用Python进行时间序列分析
  6. 计算矩阵中全1子矩阵的个数
  7. mysql 复制功能_从MySQL复制功能中得到的一举三得实惠分析
  8. 安装erlang没有bin文件夹_RabbitMQ安装教程
  9. 【2020团体程序设计天梯赛】L1部分(PTA,L1-065到L1-072)题解代码
  10. 解析button和input type=”button”的区别
  11. [Selenium]怎样验证页面是否有无变化
  12. 进展-Silverlight5、 windows phone 7、pc 三栖引擎,2D核心已完成
  13. tensorflow2 unet加载自己的图像进行训练
  14. 蒸汽密度的计算公式_蒸汽密度计算公式.doc
  15. 使用原始武器的现代战争
  16. python (ploit)
  17. IEEE 写作指南(2022)
  18. 前端资料整理--持续更新中
  19. 狂神说Java-Redis笔记
  20. Oracle中ltrim的用法

热门文章

  1. php 的常量能不能删除,php能删除常量吗
  2. SQL SERVER 2008安全配置
  3. 淘宝技术架构从1.0到4.0的演变
  4. Ukiyo-e faces dataset 浮世绘面孔数据集
  5. 音视频技术开发周刊(第122期)
  6. LiveVideoStack线上分享第三季(六):深度学习与视频编码
  7. 思科表示视频资源将在2022年占据所有互联网流量的82%
  8. 实现Jitsi SFU自动关闭/启动视频层
  9. 腾讯牌番茄,新鲜上市!
  10. ​KDD 2019 | 用户视角看世界:腾讯提出ConcepT概念挖掘系统,助力推荐搜索