博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android中关于JNI 的学习(一)对于JNIEnv的一些认识
阅读量:4921 次
发布时间:2019-06-11

本文共 2545 字,大约阅读时间需要 8 分钟。

一个简单的样例让我们初步地了解JNI的作用。可是关于JNI中的一些概念还是需要了解清楚,才干够更好的去利用它来实现我们想要做的事情。

那么C++和Java之间的是怎样通过JNI来进行互相调用的呢?

我们知道,在Android中,当Java文件被编译成dex文件之后。会由类载入器载入到Dalvik VM(DVM)中,由DVM来进行解释,翻译成机器语言之后。才干由机器来执行。

而对于C/C++来说,其源码经由Android提供的NDK工具包,能够编译成可执行动态库(即.so文件),之后,Java和C++之间就能够进行通讯了。

那么,在这里,能够想像,Java的Dex字节码和C/C++的so库肯定是同一时候执行在一个DVM之中,它们是共同使用一个进程空间的,否则,它们怎么彼此沟通呢?

所以在这里,一个关键的中间区域就是Dalvik VM。而对于C/C++,当它们也被载入进DVM之后。由C/C++实现的函数方法等都会被载入在DVM中的函数表中。

假设想要在C/C++中调用函数。它们必需要有个东西能够让其訪问到这个虚拟机中的函数表。

而这个东西就是JNIEnv *。

当我们利用javah生成的C/C++的头文件的时候,例如以下:

JNIEXPORT jstring JNICALL Java_com_lms_jni_HwDemo_printHello  (JNIEnv *e, jobject j){        return (**e).NewStringUTF(e,"Hello from T" );}

我们能够看到这种方法有两个參数,当中第一个就是JNIEnv *,而我们在Java端定义这种方法的时候。是没有參数的。例如以下:

public native String printHello();

那么这个JNIEnv是干什么用的?
事实上从这个參数的名称就能够看到,就是指JNI的执行环境,我认为它就是对Java虚拟环境的一个引用。在Android中,就是指Dalvik VM。
參考jni.h文件里关于JNIEnv的定义,例如以下(对于C和C++,它的定义有点不一样):
struct _JNIEnv;struct _JavaVM;typedef const struct JNINativeInterface* C_JNIEnv;#if defined(__cplusplus)typedef _JNIEnv JNIEnv; //C++中JNIEnv的类型typedef _JavaVM JavaVM; #elsetypedef const struct JNINativeInterface* JNIEnv; //C中JNIEnv的类型typedef const struct JNIInvokeInterface* JavaVM;#endif
在C中,我们能够看到JNIEnv的类型就是JNINativeInterface* ,是一个指针类型,那么在C++中呢,_JNIEnv是什么样的呢?
struct _JNIEnv {    /* do not rename this; it does not seem to be entirely opaque */    const struct JNINativeInterface* functions;
而对于C++来说, _JNIEnv是一个结构体,里面包括了JNINativeInterface*的结构。

所以从这里也能够看到。对于C和C++来说,它们引用JNIEnv中的方法是有一点不一样的。

总的来说。JNIEnv,无论是C。还是C++。事实上关键都是JNINativeInterface的这个结构。

我们能够简单看一下JNINativeInterface结构的定义,例如以下:
struct JNINativeInterface {    void*       reserved0;    void*       reserved1;    void*       reserved2;    void*       reserved3;    jint        (*GetVersion)(JNIEnv *);    jclass      (*DefineClass)(JNIEnv*, const char*, jobject, const jbyte*,                        jsize);    jclass      (*FindClass)(JNIEnv*, const char*);    jmethodID   (*FromReflectedMethod)(JNIEnv*, jobject);    jfieldID    (*FromReflectedField)(JNIEnv*, jobject);    /* spec doesn't show jboolean parameter */    jobject     (*ToReflectedMethod)(JNIEnv*, jclass, jmethodID, jboolean);
能够看到在它当中定义了非常多的函数指针,而通过这些定义,JNI层事实上就获得了对DVM的引用,通过定义的这些函数指针,能够定位到虚拟机中的 JNI 函数表。从而实现JNI层在DVM中的函数调用。

所以,能够这样理解,事实上JNIEnv。就是对DVM执行环境中C/C++函数的一个引用。而也正由于此,当C/C++想要在DVM中调用函数的时候。由于其是在DVM的环境中。所以它们必须通过JNIEnv* 这个參数来获得这些方法,之后才干够使用。
那么这个JNIEnv是什么时候产生的呢?
当Android中第一个Java线程要调用本地的C/C++代码的时候。DVM就会为该线程产生一个JNIEnv*的指针。而每个线程在和C/C++互相调用的时候。其相应的JNIEnv 也是相互独立。

嗯,结束。

转载于:https://www.cnblogs.com/xfgnongmin/p/10715116.html

你可能感兴趣的文章
(第3篇)HDFS是什么?HDFS适合做什么?我们应该怎样操作HDFS系统?
查看>>
隐藏 DataGrid 中 DataSource 为 DataTable 的 DataColumn (Visual C#)
查看>>
【译 】Solr in Action 第一章
查看>>
计算几何初步模板
查看>>
POJ 数据结构(2)
查看>>
HDU 3869 Color the Simple Cycle (Polya计数法)
查看>>
String字符串常用方法
查看>>
猴子们的研究
查看>>
[Python]小甲鱼Python视频第027课(集合:在我的世界里,你就是唯一)课后题及参考解答...
查看>>
sed
查看>>
关系数据库-----SQL标准语言
查看>>
java设计模式----中介模式
查看>>
常用通用JS函数
查看>>
第一章 读后心得体会
查看>>
windows下命令行cmder工具
查看>>
【深度学习大讲堂】首期第一讲:人工智能的ABCDE 第二部分:简谈当前AI技术与发展趋势...
查看>>
pandas 3 设置值
查看>>
pip无法更新
查看>>
vue-12-element组件库
查看>>
尝试读取或写入受保护的内存。这通常指示其他内存已损坏。
查看>>