2018-05-01

作者: java基础教程 | 来源:发表于2018-05-01 20:09 被阅读10次

关于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);

相关文章

网友评论

    本文标题:2018-05-01

    本文链接:https://www.haomeiwen.com/subject/svsfrftx.html