zoukankan      html  css  js  c++  java
  • Java四种引用类型

    引用与对象之间的关系?

    每种编程语言都有自己操作内存中元素的方式,例如在 C 和 C++ 里是通过指针,而在 Java 中则是通过“引用”。
    在 Java 中一切都被视为了对象,但是我们操作的标识符实际上是对象的一个引用(reference)。

    //创建一个引用,引用可以独立存在,并不一定需要与一个对象关联
    String s;
    

    通过将这个叫“引用”的标识符指向某个对象,之后便可以通过这个引用来实现操作对象了。

    String str = new String("abc");
    System.out.println(str.toString());
    

    Java 中的垃圾回收机制在判断是否回收某个对象的时候,都需要依据“引用”这个概念。
    在不同垃圾回收算法中,对引用的判断方式有所不同:

    • 引用计数法:为每个对象添加一个引用计数器,每当有一个引用指向它时,计数器就加1,当引用失效时,计数器就减1,当计数器为0时,则认为该对象可以被回收(目前在Java中已经弃用这种方式了)。
    • 可达性分析算法:从一个被称为 GC Roots 的对象开始向下搜索,如果一个对象到GC Roots没有任何引用链相连时,则说明此对象不可用。

    ​ 在 JDK1.2 之前,Java中的定义很传统:如果 reference 类型的数据中存储的数值代表的是另外一块内存的起始地址,就称为这块内存代表着一个引用。一个对象只有“已被引用”和"未被引用"两种状态,这将无法描述某些特殊情况下的对象,比如,当内存充足时需要保留,而内存紧张时才需要被抛弃的一类对象。

    四种引用类型

    所以在 JDK.1.2 之后,Java 对引用的概念进行了扩充,将引用分为了:(默认)强引用(Strong Reference)、软引用(Soft Reference)、弱引用(Weak Reference)、虚引用(Phantom Reference)4 种,这 4 种引用的强度依次减弱。

    一、强引用

    Java中默认声明的就是强引用,比如:

    Object obj = new Object(); //只要obj还指向Object对象,Object对象就不会被回收
    obj = null;  //将obj强引用手动置为null,可被jvm回收
    

    只要强引用存在,垃圾回收器将永远不会回收被引用的对象,一旦内存不足时,JVM会直接抛出OutOfMemoryError,不会去回收强引用数据类型的对象。如果想中断强引用与对象之间的联系,可以显示的将强引用赋值为null,这样一来,JVM就可以适时的回收对象了

    二、软引用

    软引用是用来描述一些非必需但仍有用的对象。在内存足够的时候,软引用对象不会被回收,只有在内存不足时,系统则会回收软引用对象,如果回收了软引用对象之后仍然没有足够的内存,才会抛出内存溢出异常。这种特性常常被用来实现缓存技术,比如网页缓存,图片缓存等。
    在 JDK1.2 之后,用java.lang.ref.SoftReference类来表示软引用。

    下面以一个例子来进一步说明强引用和软引用的区别:
    在运行下面的Java代码之前,需要先配置参数 -Xms2M -Xmx3M,将 JVM 的初始内存设为2M,最大可用内存为 3M。

    首先先来测试一下强引用,在限制了 JVM 内存的前提下,下面的代码运行正常

    public class TestOOM {
    	public static void main(String[] args) {
    		 testStrongReference();
    	}
    	private static void testStrongReference() {
    		// 当 new byte为 1M 时,程序运行正常
    		byte[] buff = new byte[1024 * 1024 * 1];
    	}
    }
    

    但是如果我们改为

    public class TestOOM {
    	public static void main(String[] args) {
    		 testStrongReference();
    	}
    	private static void testStrongReference() {
    		// 替换为创建一个大小为 2M 的字节数组
    		byte[] buff = new byte[1024 * 1024 * 2];
    	}
    }
    

    则内存不够使用,程序直接报错,强引用并不会被回收:报出异常OOM异常

    接着来看一下软引用会有什么不一样,在下面的示例中连续创建了 10 个大小为 1M 的字节数组,并赋值给了软引用,然后循环遍历将这些对象打印出来。

    public class TestOOM {
    	private static List<Object> list = new ArrayList<>();
    	public static void main(String[] args) {
    	     testSoftReference();
    	}
    	private static void testSoftReference() {
    		for (int i = 0; i < 10; i++) {  
    			byte[] buff = new byte[1024 * 1024];  // 创建10个大小1M且强引用的数组
    			SoftReference<byte[]> sr = new SoftReference<>(buff); // 创建数组的软应用
    			list.add(sr);  // 10个软引用对象存入数组
    		}
    		
    		System.gc(); //主动通知垃圾回收,软引用对象会被清除
    		
    		for(int i=0; i < list.size(); i++){
    			Object obj = ((SoftReference) list.get(i)).get();
    			System.out.println(obj);
    		}
    	}	
    }
    

    打印结果:

    ​ 无论循环创建多少个软引用对象,打印结果总是只有最后一个对象被保留,其他的obj全都被置空回收。说明在内存不足的情况下,软引用将会被自动回收。
    ​ 值得注意的一点 , 即使有 byte[] buff 引用指向对象, 且 buff 是一个strong reference, 但是 SoftReference sr 指向的对象仍然被回收了,这是因为Java的编译器发现了在之后的代码中, buff 已经没有被使用了, 所以自动进行了优化。

    将上面示例稍微修改如下:

    	private static void testSoftReference() {
    		byte[] buff = null;   // 强引用指向null
    
    		for (int i = 0; i < 10; i++) {
    			buff = new byte[1024 * 1024];
    			SoftReference<byte[]> sr = new SoftReference<>(buff);
    			list.add(sr);
    		}
    
            System.gc(); //主动通知垃圾回收
    		
    		for(int i=0; i < list.size(); i++){
    			Object obj = ((SoftReference) list.get(i)).get();
    			System.out.println(obj);
    		}
    
    		System.out.println("buff: " + buff.toString());
    	}
    

    则 buff 会因为强引用的存在,而无法被垃圾回收,从而抛出OOM的错误。

    如果一个对象惟一剩下的引用是软引用,那么该对象是软可及的(softly reachable),此时垃圾收集器并不像尽量收集弱可及对象一样收集软可及对象。相反,它只在真正 “需要” 内存时才收集软可及的对象。

    三,弱引用

    弱引用的引用强度比软引用要更弱一些,无论内存是否足够,只要 JVM 开始进行垃圾回收,那些被弱引用关联的对象都会被回收。在 JDK1.2 之后,用 java.lang.ref.WeakReference 来表示弱引用。
    我们以与软引用同样的方式来测试一下弱引用:

    	private static void testWeakReference() {
    		for (int i = 0; i < 10; i++) {
    			byte[] buff = new byte[1024 * 1024];
    			WeakReference<byte[]> sr = new WeakReference<>(buff);
    			list.add(sr);
    		}
    		
    		System.gc(); //主动通知垃圾回收
    		
    		for(int i=0; i < list.size(); i++){
    			Object obj = ((WeakReference) list.get(i)).get();
    			System.out.println(obj);
    		}
    	}
    

    打印结果:所有被弱引用关联的对象都被垃圾回收

    四,虚引用

    虚引用是最弱的一种引用关系,如果一个对象仅持有虚引用,那么它就和没有任何引用一样,它随时可能会被回收,在 JDK1.2 之后,用 PhantomReference 类来表示,通过查看这个类的源码,发现它只有一个构造函数和一个 get() 方法,而且它的 get() 方法仅仅是返回一个null,也就是说将永远无法通过虚引用来获取对象,虚引用必须要和 ReferenceQueue 引用队列一起使用。

    public class PhantomReference<T> extends Reference<T> {
        public T get() {
            return null;
        }
        public PhantomReference(T referent, ReferenceQueue<? super T> q) {
            super(referent, q);
        }
    }
    

    那么传入它的构造方法中的 ReferenceQueue 又是如何使用的呢?

    五,引用队列(ReferenceQueue)

    引用队列可以与软引用、弱引用以及虚引用一起配合使用,当垃圾回收器准备回收一个对象时,如果发现它还有引用,那么就会在回收对象之前,把这个引用加入到与之关联的引用队列中去。程序可以通过判断引用队列中是否已经加入了引用,来判断被引用的对象是否将要被垃圾回收,这样就可以在对象被回收之前采取一些必要的措施。

    参考博客

    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利
  • 相关阅读:
    Markdown入门
    HTTP协议 keep-alive连接 与 BS(firefox-thttpd)实验
    emoji探寻之路
    JavaScript实现绑定DOM的定时器插件
    C语言 str2bin 和 bin2str 实现
    LUA OOP 单例模式实现的 一个 方案
    LUA OOP编程实现方法
    以一则LUA实例说明敏捷开发中“分离构造和使用”原则
    HTML 中按钮作为form表单元素提交特性两则 --- 参HTML考标准分析
    分享:一款前端布局工具(alloydesigner)
  • 原文地址:https://www.cnblogs.com/hhddd-1024/p/14592211.html
Copyright © 2011-2022 走看看