原 面试题(五)强引用、软引用、弱引用、虚引用分别是什么?
						
							版权声明:本文为博主原创文章,请尊重他人的劳动成果,转载请附上原文出处链接和本声明。
						
						
							本文链接:https://www.91mszl.com/zhangwuji/article/details/1165
						
						
						

(1)当内存不足,JVM开始垃圾回收,对于强引用的对象,就算是出现了OOM也不会对该对象进行回收,死都不收。
(2)强引用是我们最常见的普通对象引用,只要还有强引用指向一个对象,就能表明对象还“活着”,垃圾收集器不会碰这种对象,在java中最常见的就是强引用。把一个对象赋给一个引用变量,这个引用变量就是一个强引用。当一个对象被强引用变量引用时,它处于可达状态,它是不可能被垃圾回收机制回收的,即使该对象以后永远都不会被用到JVM也不会回收。因此强引用是造成java内存泄漏的主要原因之一。
(3)对于一个普通的对象,如果没有其他的引用关系,只要超过了引用的作用域或者显示的将相应(强)引用赋值为null,一般认为就是可以被垃圾收集的(当然具体回收时机还是要看垃圾收集策略)。
强引用演示代码
package com.mszl.gc;
/**
 * 功能:强引用
 * 备注:更多资料请访问 http://www.91mszl.com
 * @author bobo teacher
 */
public class ReferenceDemo1 {
	
	public static void main(String[] args) {
		Object obj1=new Object(); // 这样的写法就是强引用
		Object obj2=obj1; // obj2 引用赋值
		obj1=null;
		System.gc();
		
		System.out.println(obj1);
		System.out.println(obj2);
	}
	
	
}执行结果
null
java.lang.Object@1db9742二:软引用(在JVM内存充足的情况下不进行回收,JVM内存不够的时候进行回收)。
(1)软引用是一种相对强引用弱化了一些引用,需要用java.lang.ref.SoftReference类实现,可以让对象豁免一些垃圾收集。
(2)对于只有软引用的对象来说,当系统内存充足时不会被回收,当系统内存不足时会被回收。
(3)软引用通常在堆内存敏感的程序中,比如高速缓存就有用到软引用,内存够用的时候就保留,不够用就回收。
软引用内存充足的情况演示,代码如下
package com.mszl.gc;
import java.lang.ref.SoftReference;
/**
 * 功能:软引用
 * 备注:更多资料请访问 http://www.91mszl.com
 * @author bobo teacher
 */
public class ReferenceDemo2 {
	// 内存充足
	public static void memoryEnough(){
		Object obj=new Object();
		SoftReference<Object> sf=new SoftReference<Object>(obj);
		System.out.println(obj);
		System.out.println(sf.get());
		
		obj=null;
		System.gc();
		
		System.out.println(obj);
		System.out.println(sf.get());
	}
	
	public static void main(String[] args) {
		memoryEnough();
	}
}执行结果
java.lang.Object@1db9742
java.lang.Object@1db9742
null
java.lang.Object@1db9742软引用内存不足的情况演示,代码如下
package com.mszl.gc;
import java.lang.ref.SoftReference;
/**
 * 功能:软引用
 * 备注:更多资料请访问 http://www.91mszl.com
 * @author bobo teacher
 */
public class ReferenceDemo2 {
	// 内存不足:我们eclipse设置为 -Xms10m -Xmx10m
	public static void memoryNotEnough(){
		Object obj=new Object();
		SoftReference<Object> sf=new SoftReference<Object>(obj);
		System.out.println(obj);
		System.out.println(sf.get());
		
		obj=null;
		try{
			byte[] bt=new byte[30 * 1024 * 1024];
		} catch(Exception ex){
			ex.printStackTrace();
		} finally{
			System.out.println(obj);
			System.out.println(sf.get());
		}
	}
	
	public static void main(String[] args) {
		memoryNotEnough();
	}
}执行结果
java.lang.Object@1db9742
java.lang.Object@1db9742
null
null
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
	at com.mszl.gc.ReferenceDemo2.memoryNotEnough(ReferenceDemo2.java:21)
	at com.mszl.gc.ReferenceDemo2.main(ReferenceDemo2.java:31)弱引用演示代码
package com.mszl.gc;
import java.lang.ref.WeakReference;
/**
 * 功能:弱引用
 * 备注:更多资料请访问 http://www.91mszl.com
 * @author bobo teacher
 */
public class ReferenceDemo3 {
	
	public static void main(String[] args) {
		Object obj=new Object();
		WeakReference<Object> wr=new WeakReference<Object>(obj);
		
		System.out.println(obj);
		System.out.println(wr.get());
		
		obj=null;
		System.gc();
		
		System.out.println(" ---------------- ");
		
		System.out.println(obj);
		System.out.println(wr.get());
	}
}执行结果
java.lang.Object@128f6ee
java.lang.Object@128f6ee
 ---------------- 
null
null引用队列演示代码如下:
package com.mszl.gc;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.WeakReference;
/**
 * 功能:引用队列
 * 备注:更多资料请访问 http://www.91mszl.com
 * @author bobo teacher
 */
public class ReferenceDemo5 {
	
	public static void main(String[] args) throws InterruptedException {
		Object obj=new Object();
		ReferenceQueue<Object> queue=new ReferenceQueue<Object>(); // 引用队列
		WeakReference<Object> wf=new WeakReference<Object>(obj, queue); // 弱引用
		
		System.out.println(obj);
		System.out.println(wf.get());
		System.out.println(queue.poll());
		
		System.out.println("=============");
		obj=null;
		System.gc();
		Thread.sleep(1000);
		
		System.out.println(obj);
		System.out.println(wf.get());
		System.out.println(queue.poll());
	}
	
}执行结果
java.lang.Object@128f6ee
java.lang.Object@128f6ee
null
=============
null
null
java.lang.ref.WeakReference@166afb3(2)顾名思义,就是形同虚设,与其他几种引用都不同,虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收,它不能单独使用也不能通过它访问对象,虚引用必须和引用队列(ReferenceQueue)联合使用。
(3)虚引用的主要作用是跟踪对象被垃圾回收的状态。仅仅是提供了一种确保对象被finalize以后,做某些事情的机制。PhantomReference的get方法总是返回null,因此无法访问对应的引用对象。其意义在于说明一个对象已经进入finalization阶段,可以被gc回收,用来实现比finalization机制更灵活的回收操作。

虚引用演示代码
package com.mszl.gc;
import java.lang.ref.PhantomReference;
import java.lang.ref.ReferenceQueue;
/**
 * 功能:虚引用
 * 备注:更多资料请访问 http://www.91mszl.com
 * @author bobo teacher
 */
public class ReferenceDemo6 {
	
	public static void main(String[] args) throws InterruptedException {
		Object obj=new Object();
		ReferenceQueue<Object> queue=new ReferenceQueue<Object>(); // 引用队列
		PhantomReference<Object> pf=new PhantomReference<Object>(obj, queue); // 虚引用
		
		System.out.println(obj);
		System.out.println(pf.get());
		System.out.println(queue.poll());
		
		System.out.println("=============");
		obj=null;
		System.gc();
		Thread.sleep(1000);
		
		System.out.println(obj);
		System.out.println(pf.get());
		System.out.println(queue.poll());
	}
	
}执行结果
java.lang.Object@166afb3
null
null
=============
null
null
java.lang.ref.PhantomReference@b5dac4总结:
(1)java提供了4种引用类型,在垃圾回收的时候,都有自己各自的特点。ReferenceQueue是用来配合引用工作的,没有referenceQueue一样可以运行。
(2)创建引用的时候可以指定关联的队列,当GC释放对象内存的时候,会将引用加入到引用队列,如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用对象的内存被回收之前采取必要的行动,这相当于是一种通知机制。
(3)当关联的引用队列中有数据的时候,意味着引用指向的堆内存中的对象被回收。通过这种方式,JVM允许我们在对象被销毁后做一些我们自己想做的事情。
2019-12-06 18:31:33 阅读(1471)
名师出品,必属精品 https://www.91mszl.com