jvm将硬件、操作系统底层和应用层隔离开来,使得java成为了一种跨平台语音。java开发人员不需要关心硬件和操作系统的差异,jvm自动将class文件翻译成适合的机器语言。
与此同时,很多优秀的库,或者操作系统如Linux,是由C/C++编写的,如果能使用java调用这些库,将会提高开发效率。jni就是java和C/C++代码之间的桥梁。通过JNI,java可以调用C/C++代码。
安卓中JNI的使用方法
1.创建java的native函数。
2.使用javah自动生成对应的jni函数。
3.在生成的jni函数中实现逻辑,如调用C/C++方法,或者调用java方法。
4.使用Android.mk、Application.mk将jni函数打包成.so库。
5.在java程序中加载.so库,使用native函数。
JNI中调用java方法
1.使用JNIEnv->FindClass("java类名")获取java类
2.使用JNIEnv->GetMethodId("获取到的java类","方法名","方法签名");获取java方法
3.使用env->CallVoidMethod(jobject,"获取的java方法","输入参数");调用java方法
JNI中JNIEnv的意义
JNIEnv是一个指针,指向jvm中保存的JNI函数表。通过JNIEnv可以调用JNI函数。JNIEnv是线程私有的,每个线程有独自的JNIEnv。通过JavaVM->GetEnv()可以获得本线程的JNIEnv。
JNI中的变量声明周期
JNI中变量有三种声明周期
1.局部引用(Local Reference)
在所创建的native函数中有效,在函数结束时结束声明周期(可以被GC)。
2.全局引用(Global Reference)
可以跨方法,跨线程使用,在主动释放之前不能被GC。
3.弱全局引用(Weak Global Reference)
可以跨方法,跨线程使用,不释放也可能被GC。
JNI多线程
JNI线程用到的13三个函数
1、pthread_t :用于声明一个线程对象如:pthread_t thread;
2、pthread_creat :用于创建一个实际的线程如:pthread_create(&pthread,NULL,threadCallBack,NULL);其总共接收4个参数,第一个参数为pthread_t对象,第二个参数为线程的一些属性我们一般传NULL就行,第三个参数为线程执行的函数( void* threadCallBack(void *data) ),第四个参数是传递给线程的参数是void*类型的既可以传任意类型。
3、pthread_exit :用于退出线程如:pthread_exit(&thread),参数也可以传NULL。注:线程回调函数最后必须调用此方法,不然APP会退出(挂掉)。
4、pthread_mutex_t :用于创建线程锁对象如:pthread_mutex_t mutex;
5、pthread_mutex_init :用于初始化pthread_mutex_t锁对象如:pthread_mutex_init(&mutex, NULL);
6、pthread_mutex_destroy :用于销毁pthread_mutex_t锁对象如:pthread_mutex_destroy(&mutex);
7、pthread_cond_t :用于创建线程条件对象如:pthread_cond_t cond;
8、pthread_cond_init :用于初始化pthread_cond_t条件对象如:pthread_cond_init(&cond, NULL);
9、pthread_cond_destroy :用于销毁pthread_cond_t条件对象如:pthread_cond_destroy(&cond);
10、pthread_mutex_lock :用于上锁mutex,本线程上锁后的其他变量是不能被别的线程操作的如:pthread_mutex_lock(&mutex);
11、pthread_mutex_unlock :用于解锁mutex,解锁后的其他变量可以被其他线程操作如:pthread_mutex_unlock(&mutex);
12、pthread_cond_signal :用于发出条件信号如:pthread_cond_signal(&mutex, &cond);
13、pthread_cond_wait :用于线程阻塞等待,直到pthread_cond_signal发出条件信号后才执行退出线程阻塞执行后面的操作。
JNI中线程需要使用JNIEnv的使用,需要调用(*g_jvm)->AttachCurrentThread(g_jvm, &env, NULL),将自身绑定到JVM中。