JNA定义:

JNA:java Native Access,是SUN公司开发的基于JNI的框架。JNI使得Java可以调用原生的c或者c++代码。

JNA与JNI(Java Native Interface)的差别:

性能:JNA在性能上不如JNI。由于JNA是在JNI的基础上封装了一层。
移植性:JNA的可移植性要好于JNI,由于开发人员不须要再编写作为代理的动态链接库。
使用:JNI使用native关键字,使用一个个java方法映射原生方法,利用System.loadLibrary;JNA使用一个java借口来代表动态链接库。使用Native.loadLibrary

JNA使用环境安装:

原生代码:使用C++或者C编写原生代码,或者使用已有的原生代码。在准备在java中使用的函数或者class前注明extern “C” __declspec(dllexport)。然后打包成动态链接库dll
Java代码:下载jna.jar,https://github.com/java-native-access/jna。然后把dll文件放在详细的project以下。

JNA使用:

1. 准备dll(这部分针对刚開始学习的人。可直接浏览第2部分)
我们从生成dll到利用JNA,使用Java调用dll一步一步解说。
首先生成dll。为了学习JNA。最好我们自己动手生成dll,我是使用Dev C++(http://sourceforge.net/projects/orwelldevcpp/) 这个工具来编写c++代码的。安装Dev C++之后,我们创建一个dllproject。

然后我们创建一个.h文件以及一个.cpp文件。我们须要在.h里面define:

#if BUILDING_DLL
#define DLLIMPORT extern "C" __declspec(dllexport)
#else
#define DLLIMPORT extern "C" __declspec(dllimport)
#endif

然后我们在.h定义两个Struct以及两个函数function:

struct UserStruct{long id;wchar_t* name;int age;
};
DLLIMPORT
void sayUser(UserStruct* pUserStruct);
struct CompanyStruct{long id;wchar_t* name;UserStruct* users[100];int count;
};
DLLIMPORT
void sayCompany(CompanyStruct* pCompanyStruct);

这里须要注意,函数声明前须要加DLLIMPORT,这里DLLIMPORT等于extern “C” __declspec(dllexport),Struct前不须要做特殊的处理。
在.cpp文件里实现sayUser以及sayCompany两个函数:

void sayUser(UserStruct* pUserStruct)
{std::wcout<<L"hello:"<<pUserStruct->name<<std::endl;
}

sayUser函数的功能是把UserStruct结构体中的的name打印出来

void sayCompany(CompanyStruct* pCompanyStruct)
{std::wcout<<L"hello:"<<pCompanyStruct->name<<std::endl;for(int i=0;i<pCompanyStruct->count;i++){UserStruct* user=pCompanyStruct->users[i];sayUser(user);}
}

SayCompany函数的功能是把Company结构体中的name打印出来,同一时候把里面全部UserStruct成员的name打印出来。
然后我们在cpp文件里实现

BOOL WINAPI DllMain(HINSTANCE hinstDLL,DWORD fdwReason,LPVOID lpvReserved)
{return TRUE;
}

然后编译生成dll文件,由于我定义的project名字叫做JNATest,所以生成的dll文件叫做JNATest.dll。

2. Java使用JNA调用dll
我们须要创建一个interface,这里我们取名TestDll1,然后extends Library。我们须要把UserStruct和CompanyStruct两个结构体映射到java中,代码例如以下:

public class UserStruct extends Structure {public static class ByReference extends UserStruct implements Structure.ByReference{};public static class ByValue extends UserStruct implements Structure.ByValue{}public NativeLong id;public WString name;public int age;
}

当中ByReference指的是指针,ByValue指的是值,然后以下三个字段分别相应.h文件里UserStruct中的三个字段,当中顺序一定不能错。由于作为内存传给c函数调用,读取时依照c中结构体的顺序来读,所以顺序一定不能错。

同理CompanyStruct在java中的映射例如以下:

public class CompanyStruct extends Structure{public static class ByReference extends CompanyStruct implements Structure.ByReference{};public static class ByValue extends CompanyStruct implements Structure.ByValue{};public NativeLong id;public WString name;//须要使用toArray。由于java中的内存空间是不连续的。所以使用JNA提供的toArray方法生成连续的内存空间public UserStruct.ByReference[] users=(UserStruct.ByReference[]) new UserStruct.ByReference().toArray(100);public int count;
}

这里面有个须要特别注意的地方。否则会NULL错误。就是public UserStruct.ByReference[] users=(UserStruct.ByReference[]) new UserStruct.ByReference().toArray(100);这句话,假设改为public UserStruct.ByReference[] users=new UserStruct.ByReference[100],之后调用就会报错。原因是java的内存空间通常是不连续的,而我们须要连续的内存空间,这里须要使用JNA提供的toArray函数生成数组。
以下一段代码非常重要就是加载dll:

TestDll1 INSTANCE=(TestDll1) Native.loadLibrary("JNATest",TestDll1.class);

然后声明我们要在java中调用的原生函数:

public void sayUser(UserStruct userStruct);
public void sayCompany(CompanyStruct companyStruct);

然后我们创建一个Test.java,然后在main函数中測试能否调用原生函数。

首先測试sayUser:

UserStruct.ByReference userStruct=new UserStruct.ByReference();
userStruct.id=new NativeLong(100);
userStruct.age=30;
userStruct.name=new WString("SBY");
TestDll1.INSTANCE.sayUser(userStruct);

然后执行。会打印出hello:SBY
再測试sayCompany:

CompanyStruct.ByReference companyStruct=new CompanyStruct.ByReference();
companyStruct.id=new NativeLong(2);
companyStruct.name=new WString("hehe");
companyStruct.count=10;
UserStruct.ByReference pUserStruct=new UserStruct.ByReference();
pUserStruct.id=new NativeLong(90);
pUserStruct.age=99;
pUserStruct.name=new WString("sby");
//pUserStruct.write();
for(int i=0;i<companyStruct.count;i++){companyStruct.users[i]=pUserStruct;
}
TestDll1.INSTANCE.sayCompany(companyStruct);

会输出一个hello:hehe和10个hello:sby
有些地方会说须要pUserStruct.write()这行代码,目的是把内存固定住,而不被GC释放掉,然而在我測试的时候没有加这一行也能准确执行。预计这个JNA提供的toArray函数有关。

以上就是我的JNA学习心得。之所以会在machine learning的博客中插入这样一篇,基本的目的是大多数的machine leanring的代码都是使用c或者c++编写。而一些场景会须要编写java程序。此时我们须要使用java来调用已经写好的c或者c++函数。
以上源码传送门:http://yun.baidu.com/share/link?

shareid=2278504517&uk=3977203577

使用JNA,让java调用原生代码相关推荐

  1. Android NDK开发篇(四):Java与原生代码通信(原生方法声明与定义与数据类型)

    Java与原生代码通信涉及到原生方法声明与定义.数据类型.引用数据类型操作.NIO操作.訪问域.异常处理.原生线程 1.原生方法声明与定义 关于原生方法的声明与定义在上一篇已经讲一点了,这次具体分析一 ...

  2. java 调用window程序_Windows下Java调用可执行文件代码实例

    这篇文章主要介绍了Windows下Java调用可执行文件代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 缘起: 由于没有找到java转换文件 ...

  3. (JNI/JNA)java 调用c/c++ 动态链接库 全套操作+踩坑集锦

    第一篇文章终于写完-跨行三年,一直都是看别人的文章-今天咱终于自己写了一篇,自己总结的,希望能给你一点点帮助,如有错误,希望指出,立马改正. 0 前言 Java代码是跨平台的,其与硬件环境彻底&quo ...

  4. 01_JNI是什么,为什么使用,怎么用JNI,Cygwin环境变量配置,NDK案例(使用Java调用C代码),javah命令使用

    1 什么是JNI JNI Java本地开发接口 JNI是一个协议,这个协议用来沟通java代码和外部的本地代码(C/C++) 通过这个协议,java代码就可以调用外部的C/C++代码,外部的C/C++ ...

  5. java调用python代码

    同样的我们需要安装jython,具体的步骤如下: 1. 去 http://sourceforge.net/projects/jython/ 下载最新的jython相关的jar包. 2. 下载下来的ja ...

  6. java调用javafx_Java代码调用JavaFX的例子(大力推荐)

    在JavaFX 1.0发布之后,本人撰写的文章JavaFX和Java之间的互操作性被各网站转载.文中总结了3种从Java调用JavaFX的方法.这三种方法分别为: http://www.javafxb ...

  7. JNI实现Java调用C代码Demo AndroidStudio

    JNI(Java Native Interface)的本意是Java本地调用,它是为了方便java调用C/C++等本地代码所封装的一层接口 Android NDK(Native Development ...

  8. Java和MATLAB混合编程,Java调用MATLAB代码

    大家可能像我一样,是MATLAB 2014a或者MATLAB 2015a或者更高,因此找不到java package. 大家可能看到其他博客配置matlab 2013版本的,所以在这里就记录下MATL ...

  9. JAVA调用Matlab代码(MATLAB 2014a)

    本人考虑到用java绘制图形以及实现一些计算效率不高,实现起来有些复杂.而利用MATLAB写好相应的计算函数,然后打包成jar包供Java调用,在某些情况下会更加方便.或者有些时候会涉及到使用Java ...

最新文章

  1. Android开发环境——模拟器AVD相关内容汇总
  2. python 归并排序(详解)
  3. MSVCRTD.lib(crtexe.obj) : error LNK2019
  4. 如何识别真正的程序员
  5. 开启HDR视觉盛宴:究竟什么视频算得上HDR?
  6. extjs--combox用法
  7. 安装veket到移动硬盘NTFS分区
  8. Mac下matplotlib中文显示
  9. qtabbar设置不同宽度_透水地坪需要设置伸缩缝吗?
  10. Asp.Net母版页元素ID不一致的体现
  11. sql server外部连接
  12. 计算机网络课程设计小型企业局域网的组建,计算机网络课程设计小型企业局域网的组建.doc...
  13. native2ascii命令详解
  14. 据说《算法设计》是算法界三大圣经之一,你读这本书了没?
  15. 相似图片搜索算法介绍
  16. 台式计算机不同处理器型号,买电脑不要再被坑了,CPU型号解读
  17. java i=i++和j=i++的区别
  18. word字体设置:如何为常用字体设置快捷键
  19. v03.06 鸿蒙内核源码分析(时钟任务) | 调度的源动力从哪来 | 百篇博客分析HarmonyOS源码
  20. FX2LP cy7c68013——Slave FIFO 与FPGA通信

热门文章

  1. BlueTooth: 蓝牙基础知识进阶——链路控制操作
  2. 通过案例学功能 自定义监控功能初探
  3. ubuntu错误解决。
  4. VPLS(Virtual Private LAN Service)
  5. 帧中继简单总结(修改)
  6. dropout层的作用_循环神经网络的 Dropout
  7. 想学习编程,我是如何入坑python的?
  8. mysql注入攻击与防御word_SQL注入防御与绕过的几种姿势
  9. 如何平稳入门并掌握inux系统?
  10. C语言---写一个函数,判断一个字符串是否为另外一个字符串旋转之后的字符串