java中通过JNA调用dll
---恢复内容开始---
1. JNA简单介绍
先说JNI(Java Native Interface)吧,有过不同语言间通信经历的一般都知道,它允许Java代码和其他语言(尤其C/C++)写的代码进行交互,只要遵守调用约定即可。首先看下JNI调用C/C++的过程,注意写程序时自下而上,调用时自上而下。
可见步骤非常的多,很麻烦,使用JNI调用.dll/.so共享库都能体会到这个痛苦的过程。如果已有一个编译好的.dll/.so文件,如果使用JNI技 术调用,我们首先需要使用C语言另外写一个.dll/.so共享库,使用SUN规定的数据结构替代C语言的数据结构,调用已有的 dll/so中公布的函 数。然后再在Java中载入这个库dll/so,最后编写Java native函数作为链接库中函数的代理。经过这些繁琐的步骤才能在Java中调用 本地代码。因此,很少有Java程序员愿意编写调用dll/.so库中原生函数的java程序。这也使Java语言在客户端上乏善可陈,可以说JNI是 Java的一大弱点!
那么JNA是什么呢?
JNA(Java Native Access)是一个开源的Java框架,是Sun公司推出的一种调用本地方法的技术,是建立在经典的JNI基础之上的一个框架。之所以说它是JNI的替 代者,是因为JNA大大简化了调用本地方法的过程,使用很方便,基本上不需要脱离Java环境就可以完成。
如果要和上图做个比较,那么JNA调用C/C++的过程大致如下:
可以看到步骤减少了很多,最重要的是我们不需要重写我们的动态链接库文件,而是有直接调用的API,大大简化了我们的工作量。
JNA只需要我们写Java代码而不用写JNI或本地代码。功能相对于Windows的Platform/Invoke和Python的ctypes。
2. JNA技术原理
JNA使用一个小型的JNI库插桩程序来动态调用本地代码。开发者使用Java接口描述目标本地库的功能和结构,这使得它很容易利用本机平台的功能,而不会产生多平台配置和生成JNI代码的高开销。这样的性能、准确性和易用性显然受到很大的重视。
此外,JNA包括一个已与许多本地函数映射的平台库,以及一组简化本地访问的公用接口。
注意:
JNA是建立在JNI技术基础之上的一个Java类库,它使您可以方便地使用java直接访问动态链接库中的函数。
原来使用JNI,你必须手工用C写一个动态链接库,在C语言中映射Java的数据类型。
JNA中,它提供了一个动态的C语言编写的转发器,可以自动实现Java和C的数据类型映射,你不再需要编写C动态链接库。
也许这也意味着,使用JNA技术比使用JNI技术调用动态链接库会有些微的性能损失。但总体影响不大,因为JNA也避免了JNI的一些平台配置的开销。
3. JNA简单使用
JNA的项目已迁移至Github,目前最新版本是4.1.0,已有打包好的jar文件可供下载。
JNA把一个.dll/.so文件看做是一个Java接口,下面以一个简单的实例来说明怎么使用。
当然要从最经典的HelloWorld开始,我们调用C的printf函数打印出“HelloWorld”(官方的例子)
新建java project
然后finish
新建文件HelloWorld.java
在项目下创建lib文件夹,将jna.jar放入其中
在项目Properties->java Build Path->Add External JARs 中添加jna.jar
然后OK
编辑HelloWorld.java,并运行,结果如下:
运行程序,如果没有带参数则只打印出“Hello, World Hello jna!”,如果带了参数,则会打印出所有的参数。
下面来解释下这个程序。
(1)需要定义一个接口,继承自Library 或StdCallLibrary
默认的是继承Library ,如果动态链接库里的函数是以stdcall方式输出的,那么就继承StdCallLibrary,比如众所周知的kernel32库。比如上例中的接口定义:
public interface CLibrary extends Library { }
(2)接口内部定义
接口内部需要一个公共静态常量:INSTANCE,通过这个常量,就可以获得这个接口的实例,从而使用接口的方法,也就是调用外部dll/so的函数。
该常量通过Native.loadLibrary()这个API函数获得,该函数有2个参数:
- 第 一个参数是动态链接库dll/so的名称,但不带.dll或.so这样的后缀,这符合JNI的规范,因为带了后缀名就不可以跨操作系统平台了。搜索动态链 接库路径的顺序是:先从当前类的当前文件夹找,如果没有找到,再在工程当前文件夹下面找win32/win64文件夹,找到后搜索对应的dll文件,如果 找不到再到WINDOWS下面去搜索,再找不到就会抛异常了。比如上例中printf函数在Windows平台下所在的dll库名称是msvcrt,而在 其它平台如Linux下的so库名称是c。
- 第二个参数是本接口的Class类型。JNA通过这个Class类型,根据指定的.dll/.so文件,动态创建接口的实例。该实例由JNA通过反射自动生成。
CLibrary INSTANCE = (CLibrary) Native.loadLibrary((Platform.isWindows() ? "msvcrt" : "c"), CLibrary.class);
接口中只需要定义你要用到的函数或者公共变量,不需要的可以不定义,如上例只定义printf函数:
void printf(String format, Object... args);
注意参数和返回值的类型,应该和链接库中的函数类型保持一致。
(3)调用链接库中的函数
定义好接口后,就可以使用接口中的函数即相应dll/so中的函数了,前面说过调用方法就是通过接口中的实例进行调用,非常简单,如上例中:
CLibrary.INSTANCE.printf("Hello, World\n"); for (int i=0;i < args.length;i++) { CLibrary.INSTANCE.printf("Argument %d: %s\n", i, args[i]); }
这就是JNA使用的简单例子,可能有人认为这个例子太简单了,因为使用的是系统自带的动态链接库,应该还给出一个自己实现的库函数例子。其实我觉得这个完全没有必要,这也是JNA的方便之处,不像JNI使用用户自定义库时还得定义一大堆配置信息,对于JNA来说,使用用户自定义库与使用系统自带的库是完全一样的方法,不需要额外配置什么信息。比如我在Windows下建立一个动态库程序:
用vs创建DLL工程:
文件->新建->项目->visual c++->win32->win32控制台应用程序(win32项目也可以)
填写项目名称MyDLL->确定->下一步->DLL(附加选项 对空项目打钩)->完成。
到这里DLL工程就创建完毕了,下面新建两个文件MyDLL.cpp和MyDLL.h。
MyDLL.cpp内容如下:
然后Bulid -->Bulid MyDLL,dll文件就在debug文件夹下生成了编译成一个dll文件(比如Mydll.dll),将Mydll.dll放到工程的bin目录下,然后编写JNA程序调用即可:
然后修改代码并运行如下:
得到输出。
4. JNA技术难点
有过跨语言、跨平台开发的程序员都知道,跨平台、语言调用的难点,就是不同语言之间数据类型不一致造成的问题。绝大部分跨平台调用的失败,都是这个问题造成的。关于这一点,不论何种语言,何种技术方案,都无法解决这个问题。JNA也不例外。
上面说到接口中使用的函数必须与链接库中的函数原型保持一致,这是JNA甚至所有跨平台调用的难点,因为C/C++的类型与Java的类型是不一样的,你必须转换类型让它们保持一致,比如printf函数在C中的原型为:
void printf(const char *format, [argument]);
你不可能在Java中也这么写,Java中是没有char *指针类型的,因此const char *转到Java下就是String类型了。
这就是类型映射(Type Mappings),JNA官方给出的默认类型映射表如下:
还有很多其它的类型映射,需要的请到JNA官网查看。
另外,JNA还支持类型映射定制,比如有的Java中可能找不到对应的类型(在Windows API中可能会有很多类型,在Java中找不到其对应的类型),JNA中TypeMapper类和相关的接口就提供了这样的功能。
5. JNA能完全替代JNI吗?
这可能是大家比较关心的问题,但是遗憾的是,JNA是不能完全替代JNI的,因为有些需求还是必须求助于JNI。
使用JNI技术,不仅可以实现Java访问C函数,也可以实现C语言调用Java代码。
而JNA只能实现Java访问C函数,作为一个Java框架,自然不能实现C语言调用Java代码。此时,你还是需要使用JNI技术。
JNI是JNA的基础,是Java和C互操作的技术基础。有时候,你必须回归到基础上来。
6. 参考文献
(1)JNI的替代者—使用JNA访问Java外部功能接口
---恢复内容结束---
转载于:https://www.cnblogs.com/anywell/p/10242635.html
java中通过JNA调用dll相关推荐
- 使用Java中的jna调用dll
简介: 在我们日常编码中在与硬件通信或者与其他的c++平台通信的时候经常用java调用dll文件,下面我们介绍俩种使用jna调用dll的方式. 如果是自己部门写的dll文件,一般会有头文件,头文件中会 ...
- java library jna_Java JNA 调用dll库
JNA JNA(Java Native Access )提供一组Java工具类用于在运行期间动态访问系统本地库(native library:如Window的dll)而不需要编写任何Native/JN ...
- 使用Jna调用dll函数库(java使用jna对接硬件接口)
记录一次项目经历,新公司开发过程中遇到了硬件对接的需求,没有接触过这方面的我一想到这些就很头大,花了几天时间来专门研究一下这个.一般硬件购买后厂家都会附赠一些开发文档,有各种语言的demo,里面其实重 ...
- JNA调用dll - java集成一美IP对讲SDK(dll文件)
项目需要集成一美IP对讲机,一美SDK是dll文件,其他的dll文件集成做法也类似.记录一下 准备 将一美SDK文件的DataTran.dll文件放到系统C:\Windows\System32文件夹里 ...
- JNA调用DLL函数遇到的几个问题
最近一个JSP项目需要用到分词模块,而分词模块实用C++写成的DLL库.于是上网搜各种方法,最后选择了JNA作为JSP调用DLL的工具. JNA(Java Native Access )提供一组Jav ...
- Java中使用JNA实现全局监听Linux键盘事件
title: Java中使用JNA实现全局监听Linux键盘事件 date: 2019-05-03 19:08:00 Java中使用JNA实现全局监听Linux键盘事件 用JNA实现的键盘监听,在Wi ...
- Java中使用JNA实现全局监听Windows键盘事件
title: Java中使用JNA实现全局监听Windows键盘事件 date: 2019-05-02 21:55:00 Java中使用JNA实现全局监听Windows键盘事件 前言: 一直打算做一个 ...
- java中没有直接调用matlab的接口函数
java中没有直接调用matlab的接口函数,在通常的情况下我们是通过借用C++作为中间过程来实现的,使用C++主要是用c与matlab的结构实现c调用matlab的功能,将这些功能函数封装成dll文 ...
- Java中命令行调用大坑
Java中命令行调用大坑 背景 我司有一个查询服务接口机,QPS大概40~50,调用方式是Java调用Shell命令行的方式,核心代码如下: Process ps = Runtime.getRunti ...
最新文章
- PHP获取时间排除周六、周日的两个方法
- 警惕!银行风控模型或将“摇身一变”,成为风险缔造者
- LeetCode Single Number II(位操作)
- c++字符串大小比较可以用来干什么?
- [转]tomcat部署与Context
- Taro+react开发(84):taro路由跳转
- 正则表达式基础知识,持续更新…
- oracle-01122,oracle ORA-01200ORA-01110ORA-01122
- HDFS 读取、写入、遍历文件夹获取文件全路径、append
- 使用原生javaScript创建ul和li对象以及操作
- 六石管理学:头目们为什么要忽略产品质量
- 快乐的实现单独页面横屏
- 贴片钽电容封封装及规格和参数资料
- 树莓派4使用CSI摄像头
- httpwatch初级使用
- python 切割图_python切割图片的示例
- 雷军内部信:米家有品拆分 做独立电商
- 如何使用谷歌浏览器把网页保存为PNG图片
- 洛谷 P2350 [HAOI2012]外星人(素因子分解,欧拉函数)
- Autolayout的一点理解
热门文章
- iconfont 图标宽高出问题_细节猎人04期 | PPT图标的分类和使用注意事项 上篇
- jsoup 获取html中body内容_python爬虫之下载盗墓笔记(bs4解析HTML)
- JSON字符串转为指定实体类对象
- java批量删除接口怎么定义_教你在Java接口中定义方法
- linux 实时 网口 速率_Linux 命令行网络端口速度测试
- eclipse 设置workspace编码格式
- spring boot security学习
- 验证哥德巴赫猜想c语言算法,验证哥德巴赫猜想的简单优化
- centos 程序 mysql数据库文件位置,CentOS 更改MySQL数据库目录位置
- python升级pip怎么出错了_Python,开启吐槽模式,新手必看!