概述

Java 层如何调用Native层函数,大家都应该知道使用JNI(Java 本地接口)。

通过在java层声明native方法,然后遵守JNI规范命名Native函数,即可建立Java层native声明函数与Native层实现函数的关联。

另一种就是采用函数注册方式,Android Frameword层多采用这种方式,执行效率更高。

以下详细说明,两种方式的实现。

第一种方式:函数命名规范

在Android Studio 工程中,New Project 新建一个项目,将include c++ support 勾上。

pic1.png

创建出来的工程中,默认会帮你生成一个java调用native函数的示例。

//MainActivity.java

public class MainActivity extends AppCompatActivity {

static { System.loadLibrary("native-lib");}

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

final TextView tv = findViewById(R.id.sample_text);

tv.setText(stringFromJNI());

}

public native String stringFromJNI();

}

// native-lib.cpp

#include

#include

extern "C"

JNIEXPORT jstring JNICALL Java_com_felix_jnidemo_MainActivity_stringFromJNI(JNIEnv *env, jobject /* this */) {

std::string hello = "hello world from 函数命名规范方式";

return env->NewStringUTF(hello.c_str());

}

一目了然,java层 stringFromJNI 方法 与 native 层 Java_com_felix_jnidemo_MainActivity_stringFromJNI 方法,存在对应关系。

我们会发现,native层函数方法名的命名规范即是

"Java" ,包名,类名 , 方法名 以 "_" 相连

返回值 jstring 对应 java 中的String

JNICALL JNIEXPORT 表明是一个 对外暴露的JNI函数调用

特别留意一下 extern "C" ,必须加上。

如果不加上,由于C++编译器编译 cpp 文件会存在 Name Mangling (命名重整)的问题,Java_com_felix_jnidemo_MainActivity_stringFromJNI 会被C++编译器重整为另一个函数名 xxyyzzaabbcc (我随便起的名),以确保方法的独一无二性。所以java层通过调用 stringFromJNI 时,虚拟机按照默认的命名规范,找不到对应的 native实现函数,从而导致应用崩溃。

而加上 extern "C",则是通知C++编译器按照 C 的编译方式来生成函数名(即函数名保持不变)

第二种方式:函数注册方式

这种方式,写的代码稍微多点,但好处很明显,函数映射关系配置灵活,执行效率要比第一种方式高。

先要明白一个概念:

System.loadLibrary 加载动态库后,进入动态库后,会首先执行 JNI_OnLoad 这个方法,所以我们可以实现这个方法,在这个方法中注册java层与native层的函数对应关系。

具体实现流程

首先在java层中新增一个native 声明函数

public native String stringFromJNI2();

在native层提供对应的实现方法,这次我们不采用默认的命名方式

jstring stringFromJNI2(JNIEnv *env, jobject) {

std::string hello = "hello world from 函数注册方式";

return env->NewStringUTF(hello.c_str());

}

3.在Native层实现 JNI_OnLoad 方法,在这个方法中注册函数的对应关系

static int registerNativeMethods(JNIEnv *, const char *, JNINativeMethod *, int);

static int registerNatives(JNIEnv *);

jint JNI_OnLoad(JavaVM *vm, void *reserved) {

jint result = -1;

JNIEnv *env = NULL;

if (vm->GetEnv((void **) &env, JNI_VERSION_1_4)) {

goto fail;

}

// 在这里注册函数的对应关系

if (registerNatives(env) != JNI_TRUE) {

goto fail;

}

result = JNI_VERSION_1_4;

fail:

return result;

}

static JNINativeMethod gMethods[] = {{"stringFromJNI2", "()Ljava/lang/String;", (void *) stringFromJNI2}};

static int registerNativeMethods(JNIEnv *env, const char *className, JNINativeMethod *gMethods, int numMethods) {

jclass clazz;

clazz = env->FindClass(className);

if (clazz == NULL) {

return JNI_FALSE;

}

// 关键代码,在JNIEnv中 注册函数的对应关系

if (env->RegisterNatives(clazz, gMethods, numMethods) < 0) {

return JNI_FALSE;

}

return JNI_TRUE;

}

static int registerNatives(JNIEnv *env) {

if (!registerNativeMethods(env, "com/felix/jnidemo/MainActivity", gMethods, sizeof(gMethods) / sizeof(gMethods[0]))) {

return JNI_FALSE;

}

return JNI_TRUE;

}

上述代码实现中,核心代码就一句

env->RegisterNatives(clazz, gMethods, numMethods)

在这里注册了函数的对应关系,其它的都是围绕这句代码展开的。

//函数对应关系数组

static JNINativeMethod gMethods[] = {{"stringFromJNI2", "()Ljava/lang/String;", (void *) stringFromJNI2}};

JNINativeMethod 是存储映射关系的一个结构体,第一个元素是java的方法名,第二个元素是java方法对应的方法签名,第三个是native实现函数的函数指针。

下面附带一张 JNI类型签名规则表

Java类型

类型签名

boolean

Z

byte

B

char

C

long

J

float

F

double

D

short

S

int

I

L全限定类名;

数组

[元素类型签名

两种方式的比较

传统的JNI编程方式,符合JNI规范,但其缺点也明显:

方法名固定,不能灵活配置,稍不注意写错了便会出错,编译时无法发现错误

虚拟机在so库中搜索定位Native实现方法,效率有一定影响,通过注册函数可以回避这个问题

java native函数库_Java 层调用 Native 层函数的两种方式相关推荐

  1. java 线程同时启动_java多个线程同时启动的两种方式

    [背景]今天遇到一个并发问题,为了在开发环境复现这个bug,需要让多个线程同时执行到某条语句. [解决方案] java1.5的concurrent包下的CyclicBarrier 和 CountDow ...

  2. [Java中实现Excel表导入导出]基于easy-poi和EasyExcel两种方式实现

    第一种:基于easy-poi实现Excel导入导出 1.导出Excel表格 第一步:在pom文件中导入依赖 <!--基于easy-poi实现Excel导入导出--><dependen ...

  3. Java并发基础01. 传统线程技术中创建线程的两种方式

    传统的线程技术中有两种创建线程的方式:一是继承Thread类,并重写run()方法:二是实现Runnable接口,覆盖接口中的run()方法,并把Runnable接口的实现扔给Thread.这两种方式 ...

  4. Java面试题:synchronized和对象的访问定位的两种方式

    说一说自己对于 synchronized 关键字的理解 ? synchronized关键字解决的是多个线程之间访问资源的同步性,synchronized关键字可以保证被它修饰的方法或者 代码块在任意时 ...

  5. java并行任务,Java 并发编程学习(五):批量并行执行任务的两种方式

    Java 并发编程学习(五):批量并行执行任务的两种方式 背景介绍 有时候我们需要执行一批相似的任务,并且要求这些任务能够并行执行.通常,我们的需求会分为两种情况: 并行执行一批任务,等待耗时最长的任 ...

  6. Java通过图片url地址获取图片base64位字符串的两种方式

    工作中遇到通过图片的url获取图片base64位的需求.一开始是用网上的方法,通过工具类Toolkit,虽然实现的代码比较简短,不过偶尔会遇到图片转成base64位不正确的情况,至今不知道为啥. 之后 ...

  7. java class 生成对象_Java反射机制(创建Class对象的三种方式)

    1:SUN提供的反射机制的类: java.lang.Class java.lang.reflect.Constructor java.lang.reflect.Field java.lang.refl ...

  8. java数组循环扩容_Java中实现数组动态扩容的两种方法

    Java中实现数组动态扩容的两种方法 java中初始化一个数组需要定义数组的容量,而在我们使用数组时往往会遇到数组容量不够的情况,此时我们就需要通过动态扩容的方式来来根据需求扩大数组的容量. 我们可以 ...

  9. java json解析 代码_Java构造和解析Json数据的两种方法详解一

    在www.json.org上公布了很多JAVA下的json构造和解析工具,其中org.json和json-lib比较简单,两者使用上差不多但还是有些区别.下面首先介绍用json-lib构造和解析Jso ...

  10. java中写入文件_java中创建、写入文件的5种方式

    在java中有很多的方法可以创建文件写文件,你是否真的认真的总结过?下面笔者就帮大家总结一下java中创建文件的五种方法. Files.newBufferedWriter(Java 8) Files. ...

最新文章

  1. ASP.NET基础教程-DataView对象的属性、方法、枚举成员
  2. linux oracle 远程exp_linux单独安装oracle客户端及exp/imp工具配置
  3. android手机界面管理系统的设计与实现(硕士学位论文).pdf,基于Android系统的手机文件管理器的设计与实现...
  4. Java并发编程(十四)并发容器类
  5. 6.Spring Cloud Alibaba教程:Sentinel流量防卫兵的介绍与基本使用
  6. 【Linux网络编程】TCP三次握手和四次挥手
  7. P4199-万径人踪灭【FFT】
  8. php概率计算_PHP指定概率算法
  9. 打油诗 看《大上海》
  10. ARM汇编中ldr伪指令和ldr指令(转载)
  11. oracle表空间使用率统计查询
  12. Atitit.数据库事务隔离级别 attilax 总结
  13. java 开源esb_五大开源ESB项目
  14. 25种提高网页加载速度的方法和技巧
  15. 帝国站长php主动推送,帝国CMS 链接主动推送插件设置帮助
  16. 360cdn能挡住cc攻击_又被CC攻击弄得心有余悸?莫怕!这里教你如何防御
  17. 林轩田《机器学习基石》(九)—— Linear regression
  18. Fastformer论文解读
  19. 面试题:Commonjs 和 Es Module区别
  20. June 18(th)

热门文章

  1. 抖音短视频运营借势热点:有哪些热点渠道,有哪些热点改编的套路。
  2. 一加Ace竞速版,手游党的开心搭档
  3. 论软件自动化测试中 QR_Code 的登录的逻辑
  4. 酷睿i5 10300h参数 i5 10300h处理器属于什么水平 i510300h相当于台式机
  5. 苹果首款无线充电宝曝光,我们要怎么去选择
  6. CLOUD01 - KVM构建及管理 virsh控制工具、镜像管理 虚拟机快建技术
  7. 杭州拱墅区委副书记、区长冯晶一行莅临利尔达参观调研
  8. 蚂蚁资深技术专家刘晓莹十年支付宝回忆录
  9. 长在火山熔岩石板地上的大米
  10. 超全软件下载网站和网页(一网一匠)