JNA 

   JNA(Java Native Access )提供一组Java工具类用于在运行期间动态访问系统本地库(native library:如Window的dll)而不需要编写任何Native/JNI代码。开发人员只要在一个java接口中描述目标native library的函数与结构,JNA将自动实现Java接口到native function的映射。

优点

   JNA可以让你像调用一般java方法一样直接调用本地方法。就和直接执行本地方法差不多,而且调用本地方法还不用额外的其他处理或者配置什么的,也不需要多余的引用或者编码,使用很方便。

JNA描述

  JNA类库使用一个很小的本地类库sub 动态的调用本地代码。程序员只需要使用一个特定的java接口描述一下将要调用的本地代码的方法的结构和一些基本属性。这样就省了为了适配多个平台而大量的配置和编译代码。因为调用的都是JNA提供的公用jar 包中的接口。

注意

  要根据dll库文件编译版本来选择jdk版本,jdk版本与编译dll的一致,同为32位或64位。

 

JNA jar包

  jna-x.x.x.jar   jna-platform-x.x.x.jar

下面我使用JNA简单做了个demo,大致包含了一下dll的调用以及各类型的参数传递。

java的pom:

<dependency>
      <groupId>net.java.dev.jna</groupId>
      <artifactId>jna</artifactId>
      <version>5.5.0</version>
    </dependency>
    <dependency>
      <groupId>net.java.dev.jna</groupId>
      <artifactId>jna-platform</artifactId>
      <version>5.5.0</version>
    </dependency>

dll中h:

#ifndef UNTITLEDDLL_H
#define UNTITLEDDLL_H
#include "untitleddll_global.h"

class UNTITLEDDLLSHARED_EXPORT Untitleddll
{   
public:
    Untitleddll();
};

 struct GeoPos{
   double lon;
   double lat;
   double elev;
};
 /****************************
函数名:MyAdd
输    入:a 累加参数a
         b 累加参数b
输    出:int 计算结果
****************************/
extern "C" UNTITLEDDLLSHARED_EXPORT int MyAdd(int a, int b);
 /****************************
函数名:calLinkInfo
输    入:posList 位置信息结构体
         length  数组指针长度
         equipParam 传入设备参数数组
         code 信道类型
         dResult 返回计算结果数组
输    出:bool 计算是否完成
****************************/
extern "C" UNTITLEDDLLSHARED_EXPORT double calLinkInfo(GeoPos*geopos,int length,double* equipParam,int code,double *dResults);

#endif // UNTITLEDDLL_H

dll的cpp:

#include "untitleddll.h"

Untitleddll::Untitleddll()
{
}

int MyAdd(int a, int b)
{
    return a + b;
}

double calLinkInfo(GeoPos*geopos,int length,double* equipParam,int code,double *dResults)
{
    double numb = 0;
    numb = equipParam[0];
    numb += equipParam[1];
    numb += equipParam[2];
    numb += equipParam[3];
    numb += equipParam[4];

    double list = geopos->lat + geopos->lon + geopos->elev;

    *dResults = 20.0;

    return list + numb;
}

java调用:

package com.exampleweb.demo.callink;

import com.sun.jna.Library;
import com.sun.jna.Native;
import com.sun.jna.Structure;
import com.sun.jna.Structure.FieldOrder;
import com.sun.jna.ptr.DoubleByReference;
import org.junit.Test;

public class CallLinkDemo {

  static String fileName = "untitleddll";

  @FieldOrder(value={"lon","lat","elev"})
  public class GeoPos extends Structure {
    //  public static class ByReference extends GeoPos implements Structure.ByReference {}
    //  public static class ByValue extends GeoPos implements Structure.ByValue{}
    //必须使用public,不然报错
    public double lon;
    public double lat;
    public double elev;
  }

  public interface JnaLibrary extends Library {

    // fileName 为 dll 名称
    JnaLibrary INSTANCE = Native.load(fileName, JnaLibrary.class);

    //MyAdd与calLinkInfo声明函数
    int MyAdd(int a, int b);
    double calLinkInfo( GeoPos posList,int length, double[] equipParam, int code, DoubleByReference dResults);
  }

  @Test
  public void TestLink() {

    int max = JnaLibrary.INSTANCE.MyAdd(100, 200);
    System.out.println(max);
    System.out.println("//////////////分割线//////////////");

    GeoPos geopos = new GeoPos();
    geopos.lon = (10.00);
    geopos.lat = (20.00);
    geopos.elev = (30.00);

    double []equipParam = new double[5];
    for (int i = 0; i < 5; i++) {
      equipParam[i] = 0.1;
    }

    int length = 10;

    DoubleByReference reference = new DoubleByReference();//用指针返回值

    double result = JnaLibrary.INSTANCE.calLinkInfo(geopos,length,equipParam,10,reference);

    System.out.println(result);
    System.out.println(reference.getValue());
  }
}

  

最终结果:

java 调用本地dll java jna调用dll_System

常见错误:

1.注意dll名称与位置,否则报错

java.lang.UnsatisfiedLinkError: 找不到指定的模块。

2.注意jdk版本与编译dll的一致,否则报错

java.lang.UnsatisfiedLinkError: %1 不是有效的 Win32 应用程序

3.定义结构体时,必须使用public,否则报错,直接赋值或者.set()都可以

java.lang.Error: Structure.getFieldOrder() on class com.exampleweb.demo.callink.CallLinkDemo$GeoPos returns names ([elev, lat, lon]) which do not match declared field names ([])

4.注意声明函数

5.注意参数类型保持一致