强引用、软引用、弱引用、虚引用(入门篇)

相信大伙一定听过强引用、弱引用、软引用、虚引用,到底什么是强引用、弱引用、软引用、虚引用????????????????

        先从引用说起,在java中什么是引用?Person p = new Person(),  p就是引用(全称:引用数据类型变量),我相信这样的例子解释更贴切,其实在java中的引用就是指针,即内存地址,new 一个 Persion的实例对象,将引用赋值给引用变量p。

 

         再扯一扯变量的名称,就像人一样,不同的人对其不同的称谓,例如,你自己,你同学叫你同学,你同事叫你同事,你炮友叫你……,走题了!!那么变量根据不同的角度分类,就有不同的名称。 比如按照作用域的不同,称变量可为:本地变量、类变量、对象变量;按照类中所在的组成结构,又可称为局部变量、实例变量、静态变量;按照数据类型上,又可称为基本数据类型变量,引用数据类型变量。而在类结构组成中,实例变量+静态变量又可称为成员变量(字段、域,其实就是field)。

 

         不同的引用,又根据其什么特性分为 强、弱、软、虚呢? 根据被GC(垃圾处理器)回收的可能性,做了一个分类,简述一下:

          强引用(Strong Reference),是最难被GC回收的,宁可虚拟机抛出异常,中断程序,也不回收强引用指向的实例对象。

          软引用 (SoftReference),在内存不足时,GC会回收软引用指向的对象

          弱引用(WeakReference),不管内存足不足,只要我GC,我都可能回收弱引用指向的对象

          虚引用(PhantomReference ),该回收就回收,无所谓了,虚引用,我随便回收你,也叫幽灵引用,其实就是相当于没有指向任何实例对象

转载部分:

虚引用主要用来跟踪对象被垃圾回收器回收的活动。虚引用与软引用和弱引用的一个区别在于:虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要被垃圾回收。如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取必要的行动。由于Object.finalize()方法的不安全性、低效性,常常使用虚引用完成对象回收前的资源释放工作。

 

1、强引用

                好,Object fk = new Object(), fk就是强引用,咱们日常最常见的就是强引用。 强引用指向的实例对象(Heap中分配的呢),就算是jvm内存不够用了(堆内存空间),就是抛出OOM(内存溢出)OutOfMemoryError,也不去回收这个对象。牛逼不牛逼。

 

2、软引用在java中的实现

                  好,强引用老看见,怎么把强引用撸成软引用? 显示大神提供了封装类,SoftReference是个泛型类

public class SoftRe {
	public static void main(String args[]) {
		String fk = "do you like me";
		SoftReference<String> soft = new SoftReference<String>(fk);
		System.out.println(soft.get());
	}
}
3、弱引用在java中的实现

                  好,让我们再撸一个弱引用看看?大神应该有封装类的呀

public class WeakRe {
	public static void main(String args[]) {
		String temp = "fk that";
		WeakReference<String> wReference = new WeakReference<String>(temp);
		System.err.println(wReference.get());
	}
}
4、虚引用在java中的实现

                   好,让我们再看看虚引用的使用,可以看到还需要一个ReferenceQueue实例对象,而当我试图获得String实例对象的时候,发现get()返回的是null,由于引用temp已经加入了队列,所以被GC后,get()返回是null,而不是获得String实例对象。


public class PhRe {
	public static void main(String args[]) {
		String temp = "hello china";
		ReferenceQueue<String> queue = new ReferenceQueue<>();
		PhantomReference<String> phReference = new PhantomReference<String>(temp, queue);
		System.out.println(phReference.get());
	}
}

总结:不同分类的引用,对于内存管理的影响很清晰,不同实例对象的命运是不同的

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值