JNI(Java Native Interface)在多线程中的运用

转载 2012年03月26日 22:40:59

引文地址:http://blog.csdn.net/hust_liuX/archive/2006/12/25/1460486.aspx

 我在这里将文章整理了一下,重新修改了部分描述和增加了一些重要的说明事项。修改文如下:

 

问题描述:

一个java对象通过JNI调用DLL中一个send()函数向服务器发送消息,不等服务器消息到来就立即返回,同时把JNI接口的指针JNIEnv *env(虚拟机环境指针),和jobject obj保存在DLL中的变量里.

一段时间后,DLL中的消息接收线程接收到服务器发来的消息,
并试图通过保存过的env和obj来调用先前的java对象的方法(相当于JAVA回调方法)来处理此消息.此时程序会突然退出(崩溃).

解决办法:

   解决此问题首先要明白造成这个问题的原因。那么崩溃的原因是什么呢?

JNI文档上有明确表述

  The JNIEnv pointer, passed as the first argument to every native method, can only be used in the thread with which it is associated. It is wrong to cache the JNIEnv interface pointer obtained from one thread, and use that pointer in another thread.

   意思就是JNIEnv指针不能直接在多线程中共享使用。上面描述的程序崩溃的原因就在这里:回调时的线程和之前保存变量的线程共享了这个JNIEnv *env指针和jobject obj变量。

在 http://java.sun.com/docs/books/jni/html/other.html#26206 提到,
JNIEnv *env指针不可为多个线程共用,但是java虚拟机的JavaVM指针是整个jvm公用的,我们可以通过JavaVM来得到当前线程的JNIEnv指针.

于是,在第一个线程A中调用:

    

  1. JavaVM* gs_jvm;  
  2. env->GetJavaVM(&gs_jvm); //来获取JavaVM指针.获取了这个指针后,将该JavaVM保存起来。  

 

 在另一个线程B里,调用

 

 

  1. JNIEnv *env;  
  2. gs_jvm->AttachCurrentThread((void **)&env, NULL);  

 

这里还必须获取那个java对象的jobject指针,因为我们要回调JAVA方法.同 JNIEnv 指针一样,jobject指针也不能在多个线程中共享. 就是说,不能直接在保存一个线程中的jobject指针到全局变量中,然后在另外一个线程中使用它.幸运的是,可以用 
 

  1. gs_object=env->NewGlobalRef(obj);//创建一个全局变量  

 

来将传入的obj(局部变量)保存到gs_object中,从而其他线程可以使用这个gs_object(全局变量)来操纵这个java对象了.


示例代码如下:

(1)java代码:Test.java:

 

 

  1. import java.io.*;  
  2. class Test implements Runnable  
  3. {  
  4.  public int value  = 0;  
  5.  static{ System.loadLibrary("Test");}  
  6.   
  7.  public native void setEnev();//本地方法   
  8.   
  9. public static void main(String args[]) throws Exception  
  10.  {  
  11.    Test t = new Test();  
  12.    t.setEnev(); //调用本地方法   
  13.   
  14.     while(true)  
  15.     {   
  16.       Thread.sleep(1000);  
  17.       System.out.println(t.value);  
  18.     }  
  19.   }  
  20. }  

 

(2) DLL代码:Test.cpp:

  1. #include "test.h"   
  2. #include<windows.h>   
  3. #include<stdio.h>   
  4. static JavaVM *gs_jvm=NULL;  
  5. static jobject gs_object=NULL;  
  6. static int gs_i=10;  
  7.   
  8. JNIEXPORT void JNICALL Java_Test_setEnev(JNIEnv *env, jobject obj)  
  9. {  
  10.     env->GetJavaVM(&gs_jvm); //保存到全局变量中JVM   
  11.     //直接赋值obj到DLL中的全局变量是不行的,应该调用以下函数:   
  12.     gs_object=env->NewGlobalRef(obj);  
  13.   
  14.  HANDLE ht=CreateThread(NULL,0,(LPTHREAD_START_ROUTINE)ThreadFun,0,NULL,NULL);  
  15. }  
  16.   
  17. void WINAPI ThreadFun(PVOID argv)//JNI中线程回调这个方法   
  18. {   
  19.  JNIEnv *env;  
  20.  gs_jvm->AttachCurrentThread((void **)&env, NULL);  
  21.  jclass cls = env->GetObjectClass(gs_object);  
  22.  jfieldID fieldPtr = env->GetFieldID(cls,"value","I");  
  23.   
  24.  while(1)  
  25.  {  
  26.     Sleep(100);  
  27.    //这里改变JAVA对象的属性值(回调JAVA)   
  28.    env->SetIntField(gs_object,fieldPtr,(jint)gs_i++);  
  29.   }  
  30. }  

 

JNI限制:

There are certain constraints that you must keep in mind when writing native methods that are to run in a multithreaded environment. By understanding and programming within these constraints, your native methods will execute safely no matter how many threads simultaneously execute a given native method. For example:

  • JNIEnv pointer is only valid in the thread associated with it. You must not pass this pointer from one thread to another, or cache and use it in multiple threads. The Java virtual machine passes a native method the same JNIEnvpointer in consecutive invocations from the same thread, but passes different JNIEnv pointers when invoking that native method from different threads. Avoid the common mistake of caching the JNIEnv pointer of one thread and using the pointer in another thread.
  • Local references are valid only in the thread that created them. You must not pass local references from one thread to another. You should always convert local references to global references whenever there is a possibility that multiple threads may use the same reference.

java native interface JNI 调用Java方法

在上一篇文章中介绍了JNI,以及java调用JNI,这篇讲一下 JNI调用java方法。       通过使用合适的JNI函数,你可以创建Java对象,get、set 静态(static)和...
  • rain_butterfly
  • rain_butterfly
  • 2014年05月06日 14:52
  • 12341

java native interface JNI 简介、基本原理

JNI是Java Native Interface的缩写,中文为JAVA本地调用。从Java1.1开始,Java Native Interface(JNI)标准成为java平台的一部分,它允许Java...
  • rain_butterfly
  • rain_butterfly
  • 2014年05月05日 15:19
  • 18113

Jni:架起 Java 和 C/C++的桥梁,Java Native Interface

前言正如标题所说,Jni(Java Native Interface) 把 Java 和 C/C++之间联系起来了。这样的话,Java可以直接调用C/C++语言编写的代码。...
  • u013390476
  • u013390476
  • 2017年01月23日 15:56
  • 914

JNI native多线程调用Java静态方法

前言: 最近在做音视频实时通信,因为回声问题要解决,而播放和音视频采集是异步处理,所以需要java在中间做音频流桥接;播放端的音视频播放和拉流是在子线程处理的,因此有了这个应用场景。1.因为env不能...
  • hongdameng
  • hongdameng
  • 2016年09月13日 12:25
  • 1629

Java Native Interface(JNI)从零开始详细教程

==================================================================================== 首先声明:这边文章是我翻译的...
  • Baniel01
  • Baniel01
  • 2016年12月22日 13:37
  • 3041

chap 17:当JNI遇到多线程--java对象如何被C++中的多个线程访问?

java中要访问C++代码时,使用JNI是唯一选择.然而,在多线程的情况下,可能出现以下问题: 问题描述: 一个java对象通过JNI调用DLL中一个send()函数向服务器发送消息,不等服务器消...
  • zvivi521
  • zvivi521
  • 2012年04月16日 08:47
  • 1056

JNI(Java Native Interface)在多线程中的运用 2

在 JNI(Java Native Interface)在多线程中的运用 一文中说到了 gs_jvm->AttachCurrentThread((void **)&env, NULL);这个方法,今天...
  • louyong0571
  • louyong0571
  • 2011年07月28日 15:29
  • 1028

JNI(Java Native Interface)在多线程中的运用

引文地址:http://blog.csdn.net/hust_liuX/archive/2006/12/25/1460486.aspx 我在这里将文章整理了一下,重新修改了部分描述和增加了一些重要的说...
  • lovingprince
  • lovingprince
  • 2008年08月19日 09:10
  • 14911

JNI官方文档翻译7-Java线程和Native线程

在写这篇文章之前,我还是先简单概括一下第七章的内容:对于java程序而言,native是我们需要访问的库。但是对于一个单纯的c程序而言,我们可以使用一个特殊的c库来创建虚拟机。刚好相反。在C程序里创建...
  • mtaxot
  • mtaxot
  • 2016年05月27日 10:25
  • 1992

Java Native Interface Specification(JNI)

Java Native Interface Specification(JNI) 使用场景: 需要的功能,标准的java不能提供有了一个用其他的语言写好的工具包,希望用java去访问它当需要高...
  • luhuajcdd
  • luhuajcdd
  • 2013年07月22日 11:12
  • 1760
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:JNI(Java Native Interface)在多线程中的运用
举报原因:
原因补充:

(最多只允许输入30个字)