文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java的JNA类型映射注意细节及使用方法

2023-06-30 03:54

关注

这篇文章主要介绍“Java的JNA类型映射注意细节及使用方法”,在日常操作中,相信很多人在Java的JNA类型映射注意细节及使用方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的JNA类型映射注意细节及使用方法”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

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 directlyconst char* getString();// Example B: Returns a wide character C string directlyconst wchar_t* getString();

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

// Mapping AString getString();// Mapping BWString 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': // double      sum += (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));

到此,关于“Java的JNA类型映射注意细节及使用方法”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯