×
广告

"放弃Jni"愉快的奔向JNA

96
最爱大包子
2017.12.06 13:02* 字数 1012

JNA的使用

使用背景

  • 在安卓开发过程中,常常会需要和C,C++等交互,这时候我们就想起了JNI,但是JNI的使用过程十分繁琐,而且容易出现各种形形色色的问题,还得封装而且问题不好定位所以对调用大量的c程序造成困扰。相比之下JNA的简易程度令人发指,你只需要一个jar包,就可以像调用一个java方法一样简单。假如我们只有一个.so文件,我们使用JNI去调用,我们需要另外用C语音写一个.so的共享文件,并且得使用SUN规定的数据结构去替代C语言的数据结构,至此才能调用so文件里面公布的函数。作为搞JAVA的程序员这个过程是令人头疼的。
对比

如上图所示Jna省去了令Java程序员头疼的Jni中间键共享库的封装。官方言简意赅的描述:JNA provides Java programs easy access to native shared libraries without writing anything but Java code - no JNI or native code is required.

  • 为了更加直观的体现Jna的优势,我编了一个简单的so库,分别用JNA和Jni来调用Native方法。
    Native 方法和头文件内容如下:


    2.png
3.png

编译得到wer.so里面有add方法。

  • 现在我们使用Jni方式去调用wer里面的add方法。
  • 先去封装jni调用的中间键,代码如如图:


    4.png
5.png
6.png

在编写jni中间键的时候还要注意头文件的引用,方法的全路径,Android.mk的配置,稍有不慎在调用的时候就会引起喜闻乐见的崩溃。

  • 在项目中调用封装好的库,如下图:


    7.png

注意这块的方法全路径必须和native里面的一致。Jni必须这样,然后jna就可以不这样。

Jna的调用

  • 引入jna的jar包和对应的jnidispatch.so.然后编写jnatools 如下:


    8.png

仅仅这么一步就搞定了,激动不!!!

JNA简介

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


    type.png

JNA的使用

  • 在github上下载jna.jar和对应平台的 libjnidispatch.so ,并在项目中配置好。

懒人传送门CSDN下载也可以clone我的demo里面有项目源码

  • 根据Native的头文件愉快的编写Java代码。

  • 一路哼着小歌就把Native方法调起来了,再也不用考虑jni的路径问题,可以在任意路径下折腾啦。
    运行结果如下:


    展示 .png
  • jna中 一些常用数据类型的简单示例。

  • char *

c代码实现

int getName(char *name,char * s) {
    if(name==NULL||s==NULL){
        return -1;
    }
    int result=strcmp(name, "大包子");
    char  dabaozi[40] ="大包子最帅!";
    char other [40] = "其他人一般帅。。。";
        if (result == 0) {

        memcpy(s, dabaozi, strlen(dabaozi));
    }
    else {
        memcpy(s, other, strlen(other));
    }

    return 0;
}

java代码调用

 String s = et_name.getText().toString();
 byte [] bytename=new byte[50];
ByteBuffer call=ByteBuffer.wrap(bytename);
ByteBuffer name=ByteBuffer.wrap(new String(s).getBytes());
int re = JnaTools.INSTANCE.getName(name, call);
tv_result.setText(new String(call.array()));
  • int *

c代码实现

int getAge(int *age){
    *age=18;
    return 0;
}

java代码

String s = et_name.getText().toString();
 byte [] bytename=new byte[50];
ByteBuffer call=ByteBuffer.wrap(bytename);
ByteBuffer name=ByteBuffer.wrap(new String(s).getBytes());
int re = JnaTools.INSTANCE.getName(name, call);
tv_result.setText(new String(call.array()));
  • C结构体
    c代码实现
int getStructInfo(struct _Person *  person){

    person->age=18;
    char name[10] ="大包子";
    memcpy(person->name, name, strlen(name));
    return 0;
}

java代码调用

JnaTools._Person person=new JnaTools._Person();
JnaTools.INSTANCE.getStructInfo(person);
tv_result.setText("name="+new String(person.name)+"age="+person.age);
  • JnaTools代码
package com.bigbaozi.jna.api;

import com.sun.jna.Library;
import com.sun.jna.Native;
import com.sun.jna.Structure;

import java.nio.ByteBuffer;
import java.nio.IntBuffer;
import java.util.Arrays;
import java.util.List;

/**
 * Name: JnaTools
 * Author: dabaozi
 * Email:
 * Comment: //TODO
 * Date: 2017-11-09 14:04
 */
public interface JnaTools extends Library {

        public static final String JNA_LIBRARY_NAME = "wer";
        public static final JnaTools INSTANCE = (JnaTools) Native.loadLibrary(JnaTools.JNA_LIBRARY_NAME, JnaTools.class);
        public static class _Person extends Structure {
        public int age;
        public byte[] name = new byte[11];
        public byte[] type = new byte[20];
        public _Person() {
            super();
        }
        protected List<String> getFieldOrder() {
            return Arrays.asList("age", "name", "type");
        }
        public _Person(int age, byte name[], byte type[]) {
            super();
            this.age = age;
            if ((name.length != this.name.length))
                throw new IllegalArgumentException("Wrong array size !");
            this.name = name;
            if ((type.length != this.type.length))
                throw new IllegalArgumentException("Wrong array size !");
            this.type = type;
        }
        public static class ByReference extends _Person implements Structure.ByReference {

        };
        public static class ByValue extends _Person implements Structure.ByValue {

        };
    };
    int add(int x, int y);

    int getAge(IntBuffer age);

    int getName(ByteBuffer name, ByteBuffer s);
    int getStructInfo(JnaTools._Person Person);
}

这时候有人会说每次去写JnaTools这个里面的数据类型转换我不想干,其实我也不想干,么么哒那么便成全你吧 高不高兴!!!

哈哈.jpg

前方高能,放大招中。。。。

jnatools.png
  • 如上图所示将c的头文件复制到左边,配置好类名,然后点击JNAerate ,自动转换成我们所有的Java文件。

写在最后,作为程序员的我保存住了程序员的优良传统(语言表达能力捉急)以上文章中如有不当之处,欢迎大家狠狠的批评指正

Jna
Web note ad 1