Java 强/弱/软引用

233 阅读3分钟
原文链接: mp.weixin.qq.com

关于Java的强/软/弱引用,今天总结一下他们的区别和应用。

引用的强弱程度

根据JVM对三种引用的内存回收时机来区分的话,可以把他们按

强引用 > 软引用 > 弱引用

来排列。在JVM运行内存不足时,这三种之中最先被回收的是 弱引用,依次到最后才是强引用(不会被回收)。但是对于强引用来说,JVM在内存不足时宁可抛出 OOM,也不会随意回收强引用来释放内存。下面具体说下强引用。

强引用 Strong Reference

在最经常实例化对象的语法里,如果不指定引用类型,那么默认是强引用。

Object  Object = new Object();

分两种情况来说明一下。· 在方法中的强引用· 全局强引用

方法中的强引用

在方法内声明一个强引用对象的话,在内存中会分两部分来进行。首先引用会保存在Stack中,而引用的对象Object会存放在堆中。

上面的图说明了JVM的内存模型和各种对象存放的位置。当方法执行完后,会退出方法栈,此时引用不在,所以Object会被回收。

全局强引用

其实在JVM中没有全局变量这种概念,相对的是全局静态变量。我们可以看一个类Global在编译后的字节码,

public class Global {    Global global = new Global();    public Global(){    }    public static void main(String[] args) {      System.out.println("global");    }}

javap -c Global.Class

public class Global {  Global global;  public Global();    Code:       0: aload_0       1: invokespecial #1                  // Method java/lang/Object."<init>":()V       4: aload_0       5: new           #2                  // class Global       8: dup       9: invokespecial #3                  // Method "<init>":()V      12: putfield      #4                  // Field global:LGlobal;      15: return  public static void main(java.lang.String[]);    Code:       0: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;       3: ldc           #6                  // String global       5: invokevirtual #7                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V       8: return}

可以看出来,其实非静态的变量也是在默认构造方法中实例化的,但是静态变量就不同了。静态变量是在堆中存放引用和对象,所以全局静态引用需要在不使用时将它置为null

object = null;

软引用 SoftReference

软引用在JVM内存不足时会被回收,用这种特性,可以在一些内存敏感的场景上用软引用。比如Bitmap对象,可以用软引用

SoftReference<Bitmap> bitmap = new SoftReference<Bitmap>();

弱引用 WeakReference

弱引用有着比软引用更脆弱的生命周期。即使内存充足,但是只要被GC扫描到就会被回收

WeakReference<String> abcWeakRef = new WeakReference<String>(str);