1、JNI是什么

JNI是Java Native Interface的缩写,它提供若干的API实现Java与其他语言之间的通信。而Android Framework由基于Java语言的的Java层与基于C/C++语言的C/C++层组成,每个层中的功能模块都是以有相应的语言编写,并且两层中的大部分模块有着千丝万缕的联系。而在两层之间充当连接桥梁这一角色的就是JNI,它允许Java代码和C/C++编写的应用程序与库之间进行交互;通常在以下几种情况下使用JNI

1、注重处理速度,C/C++的处理速度要优于Java语言

2、硬件控制,硬件驱动程序通常使用C语言编写,而要是Java层能够控制硬件,需要用到JNI

3、C/C++代码的复用,一些好的C/C++模块可以被多处复用

2、在Java中调用C库函数

下面以一个例子来说明在Java代码中调用C库函数的流程

1、编写Java代码

class HelloJNI {

    /*声明本地方法,该函数在C库中实现*/
native void printHello();
native void printString(String str);
/*在静态块中加载C库,可以保证在main方法前加载完成*/
static { System.loadLibrary("./hellojni"); }
public static void main(String args[])
{
HelloJNI myJNI = new HelloJNI();
/*调用C库中实现的函数*/
myJNI.printHello();
myJNI.printString("Hello world from printstring func");
}
}

在上述代码中使用native关键字声明本地方法,告诉Java编译器,此函数由其他语言编写;在静态块中加载hellojni库,该库由C语言实现,

如果是在Linux系统下则会加载libhellojni.so,如果在Windows系统下则会加载hellojni.dll;(本文以Linux系统为测试环境)

2、编译Java代码

javac HelloJNI.java

编译Java代码很简单,只要配置好JDK就可以完成编译,需要注意的是此时编译通过,但如果运行的话,由于没有实现本地函数,所以会抛出找不到函数的异常

3、生成C头文件

当Java调用本地函数printHello或者printString时并非直接映射到C语言的printHello或者printString函数,而是有一套自己的映射方法,使用如下命令即可生成C函数的头文件
javap HelloJni
执行完成后生成HelloJni.h如下
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class HelloJNI */
#ifndef _Included_HelloJNI
#define _Included_HelloJNI
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: HelloJNI
* Method: printHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_HelloJNI_printHello(JNIEnv *, jobject);
/*
* Class: HelloJNI
* Method: printString
* Signature: (Ljava/lang/String;)V
*/
JNIEXPORT void JNICALL Java_HelloJNI_printString(JNIEnv *, jobject, jstring);
#ifdef __cplusplus
}
#endif
#endif
可以看到生成的函数原型并非与Java代码调用的函数一致,函数有JNIEXPORT和JNICALL两个关键字声明,这两个关键词是必须的,有了他们JNI才能正常调用函数;而通过观察函数名称,我们可以知道其命名方式是"Java_类名_本地方法名"; 再看参数,可知JNIEnv*和jobject是本地函数的共同参数,第一个参数是JNI接口的直接,用来调用JNI提供的基本函数集;第二个参数中保存着调用本地方法的对象的一个引用,上例中的jobject中保存的对象myJNI的引用,其他的参数根据Java代码的本地方法的调用生成的 、编写C/C++代码
把上一步骤生成的HelloJni.h头文件include进来,实现其声明的函数即可,编写hellojni.c如下 #include "HelloJNI.h"
#include <stdio.h>
/*
* Class: HelloJNI
* Method: printHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_HelloJNI_printHello(JNIEnv *env, jobject obj)
{
printf("Hello World!\n");
return;
}
/*
* Class: HelloJNI
* Method: printString
* Signature: (Ljava/lang/String;)V
*/
JNIEXPORT void JNICALL Java_HelloJNI_printString(JNIEnv *env, jobject obj, jstring string)
{
/*JNI提供的基本函数集,将jstring转化成char *类型*/
const char *str = (*env)->GetStringUTFChars(env, string, );
printf("%s! \n" ,str);
return ;
}

5、生成C动态链接库

gcc -fPIC -shared -o libhellojni.so hellojni.c -I$JAVA_HOME/include

其中JAVA_HOME已经配置到环境变量中,表示JDK安装的目录,需要指定其中的include目录使用jni.h头文件

6、运行Java程序

此时执行java HelloJni会提供找不到hellojni库,这是由于在加载C库的时候在默认目录中没有找到libhellojni.so库,只需将该库复制到/usr/lib/下再次执行

xlzh@cmos:~/code/jni/simpleJNI$ java HelloJNI
Hello World!
Hello world from printstring func!

3、调用JNI函数

上图来自<Android框架揭秘>

由上图可知此示例程序有JniFuncMain类、JniTest类和libjnifunc.so(linux系统)组成,此示例有Java和C代码混合而成。

JniFuncMain类:

public class JniFuncMain
{
private static int staticIntField = ;
/*加载libjnifunc.so库*/
static { System.loadLibrary("jnifunc"); }
/*使用static关键字声明本地方法,再C库中实现*/
public static native JniTest createJniObject();
public static void main(String[] args) {
System.out.println("[Java] createJniObject() call native method");
/*调用C库的createJniObject,得到JniTest对象,注意不是用new*/
JniTest jniObj = createJniObject();
/*利用JniTest对象调用JniTest中的方法*/
jniObj.callTest();
}
}

此例中与上例不同的是本地方法返回了一个JniTest类的对象的引用,这样就可以在JniFuncMain类中调用JniTest类的方法。

JniTest类

class JniTest {
private int intField;
public JniTest(int num)
{
intField = num;
System.out.println("[Java] call JniTest: intFiled" + intField);
}
public int callByNative(int num)
{
System.out.println("[Java] JniTest 对象的 callByNative(" + num + ")调用");
return num;
}
public void callTest()
{
System.out.println("[Java] JniTest对象的callTest() 方法调用: intField = " + intField);
}
}

JniTest类提供两个方法供JniFuncMain类和C库函数调用

JniFuncMain.h

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class JniFuncMain */
#ifndef _Included_JniFuncMain
#define _Included_JniFuncMain
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: JniFuncMain
* Method: createJniObject
* Signature: ()LJniTest;
*/
JNIEXPORT jobject JNICALL Java_JniFuncMain_createJniObject
(JNIEnv *, jclass);
#ifdef __cplusplus
}
#endif
#endif

使用javah JniFuncMain生成头文件,需要注意的是第二个参数是jclass,而不是jobject,这是由于该本地方法在JniFuncMain类中声明的是static方法,所以第二个参数表示的该类的应用,而不需要对象的引用

jnifunc.cpp

#include "JniFuncMain.h"
JNIEXPORT jobject JNICALL Java_JniFuncMain_createJniObject(JNIEnv *env, jclass clazz)
{
jclass targetClass;
jmethodID mid;
jobject newObject;
jstring hellostr;
jfieldID fid;
jint staticIntField;
jint result;
/*获取JniFuncMain类的staticField变量值*/
fid = env->GetStaticFieldID(clazz, "staticIntField", "I");
staticIntField = env->GetStaticIntField(clazz, fid);
printf("[CPP] 获取 JniFuncMain类的staticIntField 值\n");
printf(" JniFuncMain.staticIntField = %d\n", staticIntField);
/*查找生成对象的类*/
targetClass = env->FindClass("JniTest");
/*查找构造方法*/
mid = env->GetMethodID(targetClass, "<init>", "(I)V");
/*生成JniTest对象*/
printf("[CPP] JniTest 对象生成 \n");
newObject = env->NewObject(targetClass, mid, );
/*调用对象的方法*/
mid = env->GetMethodID(targetClass, "callByNative", "(I)I");
result = env->CallIntMethod(newObject, mid, );
/*设置JniObject对象的intField值*/
fid = env->GetFieldID(targetClass, "intField", "I");
printf("[CPP] 设置JniTest对象的intField值为200\n");
env->SetIntField(newObject, fid, result);
/*返回对象引用*/
return newObject;
}
如果想在C代码中访问Java中的成员变量,就需要获取相应成员变量的ID值,成员变量的ID值保存在jfieldID类型的变量中;获取成员变量ID的JNI本地方法有两个,分别是 /* 获取Java中的静态成员变量ID
* env : JNI接口指针
* clazz : 包含成员变量的类的jclass
* name : 成员变量名称
* signature: 成员变量签名
*/
jfield GetStaticFieldID(JNIEnv *env, jclass clazz, const char *name, const char *signature);
/*获取Java中的普通成员变量ID*/
jfield GetFieldID(JNIEnv *env, jclass clazz, const char *name, const char *signature);
上述示例中要访问Java类中的静态成员变量,所以需要使用GetStaticFieldID方法,其他参数很简单,直接使用即可,而对于变量的签名,则需要借助Java反编译器javap命令,如下所示 xlzh@cmos:~/code/jni/middleJNI$ javap -s -p JniFuncMain
Compiled from "JniFuncMain.java"
public class JniFuncMain {
private static int staticIntField;
Signature: I
public JniFuncMain();
Signature: ()V
public static native JniTest createJniObject();
Signature: ()LJniTest;
public static void main(java.lang.String[]);
Signature: ([Ljava/lang/String;)V
static {};
Signature: ()V
}

可以看到,staticIntField的签名是I,将I传入第四个参数即可,其他函数中用到签名的时候可用同样的方法获取

OK,我们得到了成员变量的ID,那么如何通过成员变量的ID来获取或者设置成员变量的值呢?就需要用到以下几个JNI函数

/*
* 获取Java类中静态成员变量的值
* <jnitype> jobject,jboolean,jbyte,jchar, jshort, jint, jlong, jfloat, jdouble
* <type> Object ,Boolean ,Byte,Char , Short , Int, Long , Float , Double
* env: JNI接口指针
* jcalss: 包含成员变量的类
* jfieldID: 成员变量ID
*/
<jnitype> GetStatic<type>Field(JNIEnv *env, jcalss jclazz, jfieldID fieldID)
/*
* 获取Java类的对象中普通成员变量的值
*/
<jnitype> Get<type>Field(JNIEnv *env, jobject obj, jfieldID fieldID)
/*
* 设置Java类中静态成员变量的值
*/
<jnitype> SetStatic<type>Field(JNIEnv *env, jcalss clazz, jfieldID fieldID, <type> value)
/*
* 设置Java类的对象中普通成员变量的值
*/
<jnitype> Set<type>Field(JNIEnv *env, jobject obj, jfieldID fieldID, <type> value) 与成员变量类似, 获取和调用类中方法的JNI函数原型如下
/*获取Java类静态方法的ID*/
jmethod GetStaticMethodID(JNIEnv *env, jclass clazz, const char *name, const char *signature)
/*获取Java类的对象中普通方法的ID*/
jmethod GetMethodID(JNIEnv *env, jclass clazz, const char *name, const char *signature)
/*调用Java类中的静态方法*/
<jnitype> CallStatic<type>Method(JNIEnv *env, jclass clazz, jmethod methodID, ...)
/*调用Java类的对象中的普通方法*/
<jnitype> Call<type>Method(JNIEnv *env, jobject obj, jmethod methodID, ...)
如何获取Java类的对象呢?以示例中获取JniTest类的对象代码例,分为三步
、获取JniTest的类
、获取JniTest类的构造方法ID
、通过构造方法的ID调用和JniTest类使用NeoObject方法生成对象
对比上例中获取JniTest的对象流程,可以很清楚的进行对照

4、在C代码中运行Java类

Java类编译的字节码需要在Java虚拟机上运行,那么在C/C++中运行Java类自然也需要加载Java虚拟机;JNI为我们提供了一套Invocation API,它允许本地代码在自身内存区域内加载Java虚拟机,同样我们以实例的方式进行讲解

InvocationApiTest.java

public class InvocationApiTest {
public static void main(String[] args) {
System.out.println(args[]);
}
}

invocationApi.c

#include <jni.h>
int main(void)
{
JNIEnv *env;
JavaVM *vm;
JavaVMInitArgs vm_args;
JavaVMOption options[];
jint res;
jclass cls;
jmethodID mid;
jstring jstr;
jclass stringClass;
jobjectArray args;
/*加载虚拟机选项*/
options[].optionString = "-Djava.class.path=.";
vm_args.version = JNI_VERSION_1_6;
vm_args.options = options;
vm_args.nOptions = ;
vm_args.ignoreUnrecognized = JNI_TRUE;
/*生成虚拟机*/
res = JNI_CreateJavaVM(&vm, (void**)&env, &vm_args);
/*查找并加载类*/
cls = (*env)->FindClass(env, "InvocationApiTest");
/*获取main()方法的ID*/
mid = (*env)->GetStaticMethodID(env, cls, "main", "([Ljava/lang/String;)V");
/*生成字符串对象*/
jstr = (*env)->NewStringUTF(env, "Hello Invocation API!!");
stringClass = (*env)->FindClass(env, "java/lang/String");
args = (*env)->NewObjectArray(env, , stringClass, jstr); /*调用main()方法*/
(*env)->CallStaticVoidMethod(env, cls, mid, args);
/*销毁虚拟机*/
(*vm)->DestroyJavaVM(vm);
}
编译允许结果如下
xlzh@cmos:~/code/jni/superJNI$ javac InvocationApiTest.java
xlzh@cmos:~/code/jni/superJNI$ sudo echo "/usr/lib/jvm/java-1.7.0-openjdk-amd64/jre/lib/amd64/jamvm" >> /etc/ld.so.conf
xlzh@cmos:~/code/jni/superJNI$ sudo ldconfig
xlzh@cmos:~/code/jni/superJNI$ gcc -o a.out invocationApi.c -I$JAVA_HOME/include -L$JAVA_HOME/jre/lib/amd64/jamvm/ -ljvm
xlzh@cmos:~/code/jni/superJNI$ ./a.out
Hello Invocation API!!
xlzh@cmos:~/code/jni/superJNI$
												

Android JNI的使用方法的更多相关文章

  1. android JNI 调用NDK方法

    @import url(http://i.cnblogs.com/Load.ashx?type=style&file=SyntaxHighlighter.css);@import url(/c ...

  2. android JNI常用添加log方法

    android JNI 打log方法 添加库支持 LOCAL_LDLIBS :=-llog -landroid 包含头文件 #include <android/log.h> #define ...

  3. 【转】Android中JNI的使用方法

    Android中JNI的使用方法 首先看一下Android平台的框架图:(网上盗用) 可以看到Android上层的Application和ApplicationFramework都是使用Java编写, ...

  4. Android JNI使用方法

    经过几天的努力终于搞定了android JNI部分,下面将我的这个小程序和大家分享一下.android JNI是连接android Java部分和C/C++部分的纽带,完整使用JNI需要Java代码和 ...

  5. Android jni中回调java的方法

    在上一篇的基础上,添加在C++代码中回调java方法. 代码如下: Demo.java 中添加callback函数, 打印一条log. package com.example.scarecrow.dy ...

  6. JNI学习笔记_Java调用C —— Android中使用的方法

    一.笔记 1.JNI(Java Native Interface),就是如何使用java去访问C/C++编写的那些库.若想深入了解JNI可以看官方文档jni.pdf.优秀博文:Android JNI知 ...

  7. android开发源代码分析--多个activity调用多个jni库的方法

    android开发源代码分析--多个activity调用多个jni库的方法 有时候,我们在开发android项目时会遇到须要调用多个native c/jni库文件,下面是本人以前实现过的方法,假设有知 ...

  8. Android JNI和NDK学习(04)--NDK调试方法(转)

    本文转自:http://www.cnblogs.com/skywang12345/archive/2013/05/23/3092812.html 本文主要介绍在ndk中添加log的方法.然后,我们就可 ...

  9. Android jni 编程(参数的传递,成员,方法的)相互访问

    package com.test.androidjni; import android.app.Activity; import android.os.Bundle; import android.u ...

随机推荐

  1. 03-01 Java运算符

    (1)算术运算符 A:+,-,*,/,%,++,-- B:+的用法 a:加法 b:正号 c:字符串连接符 C:/和%的区别 数据做除法操作的时候,/取得是商,%取得是余数 D:++和--的用法 a:他 ...

  2. Oracle RAC 环境下的连接管理(转) --- 防止原文连接失效

    崔华老师的文章!!! 这篇文章详细介绍了Oracle RAC环境下的连接管理,分别介绍了什么是 Connect Time Load Balancing.Runtime Connection Load ...

  3. android自定义控件 几种方式总结

    方式1:不继承任何组件 , 直接在代码里面调用实例化.public class ProgressDialog { private Dialog dialog; public ProgressDialo ...

  4. 线程中的定时器Timer类

    Timer 定时器 几分钟之后执行一个任务. 创建了一个定时器相当于开启了一条线程,TimerTask相当于一个线程的任务.内部使用wait/notify机制来实现的. 用法非常的简单  就足以里面的 ...

  5. (转)percona的安装、启动、停止

    原文:https://blog.csdn.net/tanliqing2010/article/details/78758878 socket=/percona/3307/data/mysql.sock ...

  6. spring cloud开发、部署注意

    一.开发时,配置服务的配置使用本地路径,不使用svn和git,因为后者每个开发人员都会修改配置,导致别人也拿到其他人修改的配置,本地配置示例如下: spring: application: name: ...

  7. 使用binlog2sql针对mysql进行数据恢复

    MySQL闪回原理与实战 DBA或开发人员,有时会误删或者误更新数据,如果是线上环境并且影响较大,就需要能快速回滚.传统恢复方法是利用备份重搭实例,再应用去除错误sql后的binlog来恢复数据.此法 ...

  8. 这个拖后腿的“in”

    问题之源 C# 7.2推出了全新的参数修饰符in,据说是能提升一定的性能,官方MSDN文档描述是: Add the in modifier to pass an argument by referen ...

  9. RESTful SOA与DDD(领域驱动设计)

    视频地址:http://www.infoq.com/presentations/RESTful-SOA-DDD 作者的一个DDD采访:http://www.informit.com/articles/ ...

  10. JavaScript -- 数组Array

    -----021-ActiveXObject.html----- <!DOCTYPE html> <html> <head> <meta http-equiv ...