使用JNA,让java调用原生代码

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

原文地址:https://www.cnblogs.com/zhchoutai/p/8450257.html

时间: 2024-10-08 06:03:27

使用JNA,让java调用原生代码的相关文章

Android NDK开发篇(五):Java与原生代码通信(数据操作)

尽管说使用NDK能够提高Android程序的运行效率,可是调用起来还是略微有点麻烦.NDK能够直接使用Java的原生数据类型,而引用类型,由于Java的引用类型的实如今NDK被屏蔽了,所以在NDK使用Java的引用类型则要做对应的处理. 一.对引用数据类型的操作 尽管Java的引用类型的实如今NDK被屏蔽了,JNI还是提供了一组API,通过JNIEnv接口指针提供原生方法改动和使用Java的引用类型. 1.字符串操作 JNI把Java的字符串当作引用来处理,在NDK中使用Java的字符串,须要相

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

Java与原生代码通信涉及到原生方法声明与定义.数据类型.引用数据类型操作.NIO操作.訪问域.异常处理.原生线程 1.原生方法声明与定义 关于原生方法的声明与定义在上一篇已经讲一点了,这次具体分析一下.依据javah自己主动生成的头文件.能够看出原生方法的声明和定义,比如: JNIEXPORT jstring JNICALL Java_com_shamoo_helljni_HelloJni_stringFromJNI(JNIEnv *, jobject); 虽然Java上的原生方法没有不论什么

Android NDK开发篇(六):Java与原生代码通信(异常处理)

一.捕获异常 异常处理是Java中的功能,在Android中使用SDK进行开发的时候经常要用到.Android原生代码在执行过程中如果遇到错误,需要检测,并抛出异常给Java层.执行原生代码出现了问题,例如使用了空指针.内存泄漏,并且没有做相应的检测盒异常抛出,APP会马上闪退,没有任何提示. JNI中的异常处理和Java的不一样.Java中的异常处理,是直接捕获,然后做相应的处理.JNI要求开发人员在异常发生之后显式实现异常处理流.例如以下例子: public class JavaClass

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

1 什么是JNI JNI Java本地开发接口 JNI是一个协议,这个协议用来沟通java代码和外部的本地代码(C/C++) 通过这个协议,java代码就可以调用外部的C/C++代码,外部的C/C++代码也可以调用Java代码. 2 为什么用JNI 1  JNI扩展了java虚拟机的能力,驱动开发(wifi-hotspot)2.3无线热点共享 2  Native code效率高,数学运算,实时渲染的游戏上,音视频处理(极品飞车),opengl,ffmpeg 3  复用代码(文件压缩,人脸识别)

JNI Java调用C代码 示例

Activity public class MainActivity extends ListActivity {     static {         System.loadLibrary("hello");// 在java代码中引入libs目录下的库函数,文件名为[libhello.so].注意,引入时的文件名要去掉前面的lib和后面的.so                 System.loadLibrary("hellocpp");     }     

使用ecilpse(Java)调用Matlab代码

1 安装java环境: http://www.oracle.com/technetwork/java/javase/downloads/index.html 下载JDK最新版本并安装,CloudSim需要运行在jdk1.6以上版本. 以jdk1.6.0_24为例,默认的安装目录为C:\Program Files\Java\jdk1.6.0_24. 设置环境变量: 新建系统变量JAVA_HOME,变量值设为JDK安装目录,即C:\Program Files\Java\jdk1.6.0_24: 在P

java调用ruby代码

问题: 最近在做一个应用的时候碰到了一个问题.客户端需要调用服务器端传回的脚本信息,然后执行.其中脚本类型包括ruby.而java中调用ruby的代码大致如下: 1 String jrubyCode="puts 'hello world'"; 2 ScriptEngineManager manager = new ScriptEngineManager(); 3 ScriptEngine engine = manager.getEngineByName("jruby"

Java8 Nashorn实现Java调用javascript代码

最近项目需要,利用java执行动态语言,首先想到的是支持js. 1.从打印Hello World开始 Java8 引入Nashorn实现javascript调用,比如使用java运行一段js实现"Hello World"打印: package com.iflytek.research.jsdemo; import javax.script.ScriptEngine; import javax.script.ScriptEngineManager; import javax.script

1.JAVA中使用JNI调用C++代码学习笔记

Java 之JNI编程1.什么是JNI? JNI:(Java Natibe Inetrface)缩写. 2.为什么要学习JNI?  Java 是跨平台的语言,但是在有些时候仍然是有需要调用本地代码 (这些代码通常是由C/C++编写的). Sun公司提供的JNI是Java平台的一个功能强大的接口.这个JNI接口提供了Java与操作系统本地代码互相调用的功能.(即java调用C++代码) 最简单的Java调用C/C++代码步骤 1.创建TestNativeCode工程,新建cn.itcast包,新建