Interacting with Java from the Native Side |
The Java platform is a multithreaded system. Because of this, native methods must be thread-safe programs. Unless you have knowledge to the contrary, such as knowing that the native method is synchronized, you must assume that there can be multiple threads of control executing a native method at any given time. Native methods therefore must not modify sensitive global variables in unprotected ways. That is, they must share and coordinate their access to variables in certain critical sections of code.Before reading this section, you should be familiar with the concepts of threads of control and multithreaded programming. Threads of Controlcovers programming with threads. In particular, the page Multithreaded Programscovers issues related to writing programs that contain multiple threads, including how to synchronize them.
Threads and JNI
- The JNI interface pointer (JNIEnv *) is only valid in the current thread. You must not pass the interface pointer from one thread to another, or cache an interface pointer and use it in multiple threads. The Java Virtual Machine will pass you the same interface pointer in consecutive invocations of a native method from the same thread. However, different threads pass different interface pointers to native methods.
You must not pass local references from one thread to another. In particular, a local reference may become invalid before the other thread has had a chance to use it. You should always convert local references to global references in situations where different threads may be using the same reference to a Java object.
Check the use of global variables carefully. Multiple threads might be accessing these global variables at the same time. Make sure you put in appropriate locks to ensure safety.
Thread Synchronization in Native Methods
The JNI provides two synchronization functions that allow you to implement synchronized blocks. In Java, you implement synchronized blocks using thesynchronized
statement. For example:synchronized (obj) { ... /* synchronized block */ ... }The Java Virtual Machine guarantees that a thread must acquire the monitor associated with a Java object
obj
before it can execute the statements in the block. Therefore, at any given time, there can be at most one thread running inside the synchronized block.Native code can perform equivalent synchronization on objects using the JNI functions
MonitorEnter
andMonitorExit
. For example:A thread must enter the monitor associated with... (*env)->MonitorEnter(env, obj); ... /* synchronized block */ (*env)->MonitorExit(env, obj); ...obj
before it can continue its execution. A thread is allowed to enter a monitor multiple times. The monitor contains a counter signaling how many times it has been entered by a given thread.MonitorEnter
increments the counter when the thread enters a monitor it has already entered.MonitorExit
decrements the counter. Other threads can enter the monitor when the counter reaches zero (0).Wait and Notify
The functionsObject.wait
,Object.notify
, andObject.notifyAll
provide another useful thread synchronization mechanism. While the JNI does not directly support these functions, a native method can always follow the JNI method call mechanism to invoke them.
Interacting with Java from the Native Side |