JNI references
JNI reference concept: Reference variables.
Reference types: local and global references (global references contain global weak references).
Function: Tells the vm when to reclaim a JNI variable in JNI.
1. Local references
Local reference, manually releasing the object via DeleteLocalRef.
Typical usage scenarios:
Access a large Java object, and when you’re done, perform complex, time-consuming operations. Too many local references are created, consuming too much memory, and these local references have no relevance to subsequent operations. Example:
Java code declaration:
// Local references
public native void localRef(a);
Copy the code
The C code looks like this, where a large number of local references are created and deleted:
JNIEXPORT jintArray JNICALL Java_com_haocai_jni_JniTest_localRef
(JNIEnv *env, jobject jobj, jint len) {
int i = 0;
for (; i < 5; i++) {// Create a Date object
jclass cls = (*env)->FindClass(env, "java/util/Date");
jmethodID constructor_mid = (*env)->GetMethodID(env, cls, "<init>"."()V");
jobject obj = (*env)->NewObject(env, cls, constructor_mid);
// Omit a lot of code here
// The jobject object is no longer used
// Notify the garbage collector to collect these objects(*env)->DeleteLocalRef(env, obj); }}Copy the code
Java test:
public static void main(String[] args) {
JniTest jniTest = new JniTest();
jniTest.localRef();
}
Copy the code
2. Global references
Main purpose: Share (can span multiple threads), manual control of memory usage.
Java provides three methods for creating, obtaining, and deleting JNI global references.
public native void createGlobalRef(a);
public native String getGlobalRef(a);
public native void deteleGlobalRef(a);
Copy the code
C code is as follows:
/ / create
JNIEXPORT void JNICALL Java_com_haocai_jni_JniTest_createGlobalRef
(JNIEnv *env, jobject jobj, jint len) {
jstring obj = (*env)->NewStringUTF(env, "jni development is powerful!");
global_str = (*env)->NewGlobalRef(env, obj);
}
/ / to get
JNIEXPORT jstring JNICALL Java_com_haocai_jni_JniTest_getGlobalRef
(JNIEnv *env, jobject jobj, jint len) {
return global_str;
}
/ / release
JNIEXPORT void JNICALL Java_com_haocai_jni_JniTest_deleteGlobalRef
(JNIEnv *env, jobject jobj, jint len) {
(*env)->DeleteGlobalRef(env, global_str);
}
Copy the code
Java test:
public static void main(String[] args) {
JniTest jniTest = new JniTest();
jniTest.createGlobalRef();
System.out.println(jniTest.getGlobalRef());
// Release after use
jniTest.deleteGlobalRef();
System.out.println("Released...");
// A null-pointer exception will be thrown if removed
System.out.println(jniTest.getGlobalRef());
}
Copy the code
// Weak global reference // saves memory, can release the referenced object when there is insufficient memory // can reference an object that is not commonly used, if NULL, temporarily create // create: NewWeakGlobalRef, destroy :DeleteGlobalWeakRef
Exception handling //1. Ensure that Java code can run //2. Remedy to ensure that C code continues to run
JNIEXPORT void JNICALL Java_com_haocai_jni_JniTest_exeception
(JNIEnv *env, jobject jobj, jint len) {
jclass cls = (*env)->GetObjectClass(env, jobj);
jfieldID fid = (*env)->GetFieldID(env, cls, "key2"."Ljava/lang/String;");
// Check for Java exceptions
jthrowable execption = (*env)->ExceptionOccurred(env);
if(execption! =NULL) {// Let the Java code continue to run
// Clear the exception information
(*env)->ExceptionClear(env);
// Remedy
fid = (*env)->GetFieldID(env, cls, "key"."Ljava/lang/String;");
}
// Get the value of the attribute
jstring jstr = (*env)->GetObjectField(env, jobj, fid);
char *str = (*env)->GetStringUTFChars(env, jstr, NULL);
Stricmp compares strings S1 and s2, regardless of case
if (_stricmp(str, "super kpioneer") != 0) {
// Consider an exception thrown for the Java layer to handle
jclass newExcCls = (*env)->FindClass(env, "java/lang/IllegalArgumentException");
(*env)->ThrowNew(env, newExcCls, "key's value is invalid!"); }}Copy the code
3. Weak global references
The method is similar to global references:
- Create a global reference with NewWeakGlobalRef.
- Delete a global reference by DeleteWeakGlobalRef.
Unlike global references, weak global references feature:
- Save memory by freeing referenced objects when running out of memory.
- You can reference an object that is not commonly used, and if it is NULL (it has been reclaimed), you can manually create a temporary object.
Like global references, weak references can be used across methods and threads. But unlike global references, which are important, weak references do not prevent the GC from reclaiming the objects it references. Weak references are a good choice when a cached reference in native code does not necessarily prevent the GC from reclaiming the object it points to. Take a look at the following code snippet:
JNIEXPORT void JNICALL Java_com_haocai_jni_JniTest_weakRef
(JNIEnv *env, jobject obj) {
static jclass clz = NULL;
if (clz == NULL) {
jclass clzLocal = (*env)->GetObjectClass(env, obj);
if (clzLocal == NULL) {
return; /* The class */ was not found
}
clz = (jclass)(*env)->NewWeakGlobalRef(env,clzLocal);
if (clz == NULL) {
return; /* Memory overflow */}}/* Use CLZ references */
if ((*env)->IsSameObject(env,clz, NULL)) {
printf("clz is recycled\n");
}
else {
printf("clz is not recycled\n"); }}Copy the code
Java test:
public native void weakRef(a);
public static void main(String[] args) {
JniTest jniTest = new JniTest();
jniTest.weakRef();
}
Copy the code
Result output:
clz is not recycled
Copy the code
When we use a weak reference, we must first determine whether the weak reference refers to an active object or an object that has already been collected by the GC. A NULL reference refers to a NULL object in the JVM and can be detected by IsSameObject to determine whether the reference was reclaimed. Env ->IsSameObject(CLZ, NULL) returns JNI_TRUE (or 1) if returned, or JNI_FALSE (or 0) otherwise.
When our native code no longer needs a weak global reference, we should also call DeleteWeakGlobalRef to free it. If we do not manually call this function to free the pointed object, the JVM will still reclaim the object to which the weak reference points, but the memory occupied by the weak reference itself in the reference table will never be reclaimed.