Java和JNI中引用(强引用,软引用, 弱引用,虚引用)

一,强引用(对象)

特征:
强引用可以直接访问目标对象。
不会被回收。
这里写图片描述

二,软引用(SoftReference类)

特征:
当 GC 根据JVM 内存的情况来回收,JVM发现内存不够的时候就会回收
释放空间的条件: JVM 发现内存不够。
在Android的Handler 和Bitmap 使用的时候 在Activity的生命周期中不会被释放 , 成为指针指向是一个空 , 内存实际没有被释放的

这里写图片描述

三, 弱引用(WeakReference)

特征:
只要系统进行了GC,无论内存是否足够,都会被回收。

这里写图片描述

四,虚引用(PhantomReference)

get()的时候是空null的值
存储的使用, 用的时候是空的null

package com.songli.ref;

import java.lang.ref.Reference;
import java.lang.ref.SoftReference;
import java.lang.ref.WeakReference;



/**
    > Author: songli
    > QQ: 2734030745
    > Mail: 15850774503@163.com
    > CSDN: http://my.csdn.net/Poisx
    > github: https://github.com/chensongpoixs
 */


public class RefSongli {
    public static class Bean {

        String name;
        Integer va;
        public Bean(String name, Integer value) {
            this.name = name;
            this.va = value;
        }

        @Override
        public String toString() {
            // TODO Auto-generated method stub
            return name +": " + va;
        }

    }

    public static void main(String[] args) {
        //强引用

        Bean bean = new Bean("songli", 67);
         Bean[] referent = new Bean[20000]; 
         for (int i=0;i<referent.length;i++){ 
             referent[i] = new Bean("mybean:" + i,100);// 抛 Exception 
         }
         System.out.println(referent[1000].toString());

        //软引用
         Reference<Bean>[] referent = new SoftReference[20000]; 
         for (int i=0;i<referent.length;i++){ 
             referent[i] = new SoftReference<Bean>(new Bean("mybean:" + i,100)); 
         } 
         System.out.println("finish");
         System.out.println(referent[499].get() + " " + 
                 referent[10000].get() + " " + 
                 referent[19999].get());// “null”
////         
        Object object = new Object();
        WeakReference<Object> weakReference = new WeakReference<Object>(object);
//      SoftReference<Object> weakReference = new SoftReference<Object>(object);
        object = null;
        System.gc();
        System.out.println(weakReference.get()); //
    }
}

总结java中引用

这里写图片描述

Jni/NDK 中引用

1,局部引用

// 定义方式多样:FindClass,NewObject,GetObjectClass,NewCharArray…. NewLocalRef()
//释放方式: 1 方法调用完JVM 会自动释放 2.DeleteLocalRef
// 不能在多线程里面使用

JNIEXPORT void JNICALL Java_com_localRef
(JNIEnv * env, jobject jobj) {
    int i = 0;
    for (i = 0; i < 5; i++)
    {
        jclass cls = (*env)->FindClass(env, "java/util/Date");
        jmethodID jmid = (*env)->GetMethodID(env, cls, "<init>", "()V");
        //创建一个Date类型的局部引用
        jobject obj = (*env)->NewObject(env, cls, jmid);
        //使用这个引用

        //释放引用
        (*env)->DeleteLocalRef(env, cls);
        (*env)->DeleteLocalRef(env, obj);

    }
}
2,全局引用

//跨线程,跨方法使用
// NewGlobalRef 是创建全局引用的唯一方法
jstring global_str;

JNIEXPORT void JNICALL Java_com_createGlobalRef
(JNIEnv * env, jobject jobj) {
    jobject obj = (*env)->NewStringUTF(env, "JNI is intersting");
    global_str = (*env)->NewGlobalRef(env, obj);
}

JNIEXPORT jstring JNICALL Java_com_getGlobalRef
(JNIEnv * env, jobject jobj) {
    return global_str;
}

JNIEXPORT void JNICALL Java_com_delGlobalRef
(JNIEnv * env, jobject jobj) {
    (*env)->DeleteGlobalRef(env, global_str);

}
3,弱全局引用

//它不会阻止GC,/跨线程,跨方法使用
jclass g_weak_cls;

JNIEXPORT jstring JNICALL Java_com_createWeakRef
(JNIEnv * env, jobject jobj) {
    jclass cls_string = (*env)->FindClass(env, "java/lang/String");
    g_weak_cls = (*env)->NewWeakGlobalRef(env, cls_string);
    return g_weak_cls;
}

2, JNI 异常处理

//JNI 异常处理

JNIEXPORT void JNICALL Java_com_exception
(JNIEnv * env, jobject jobj) {
    jclass cls = (*env)->GetObjectClass(env, jobj);
    jfieldID fid = (*env)->GetFieldID(env, cls, "key", "Ljava/lang/String;");

    //检查是否发送异常
    jthrowable ex = (*env)->ExceptionOccurred(env);
    // 判断异常是否发送
    if (ex != NULL) {
        jclass newExc;
        //清空JNI 产生的异常
        (*env)->ExceptionClear(env);
        //IllegalArgumentException
        newExc = (*env)->FindClass(env, "java/lang/IllegalArgumentException");
        if (newExc == NULL)
        {
            printf("exception\n");
            return;
        }
        //JNI 异常获取
        (*env)->ThrowNew(env, newExc, "Throw exception from JNI: GetFieldID faild ");
    }

}

缓冲区(static)

static 在C语言中有一个静态区

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值