关于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."":()V 4: aload_0 5: new #2 // class Global 8: dup 9: invokespecial #3 // Method "":()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 = new SoftReference();
弱引用 WeakReference
弱引用有着比软引用更脆弱的生命周期。
即使内存充足,但是只要被GC扫描到就会被回收
WeakReference abcWeakRef = new WeakReference(str);