关于Java的强/软/弱引用,今天总结一下他们的区别和应用。
根据JVM对三种引用的内存回收时机来区分的话,可以把他们按
强引用 > 软引用 > 弱引用
来排列。在JVM运行内存不足时,这三种之中最先被回收的是 弱引用,依次到最后才是强引用(不会被回收)。 但是对于强引用来说,JVM在内存不足时宁可抛出 OOM,也不会随意回收强引用来释放内存。 下面具体说下强引用。
在最经常实例化对象的语法里,如果不指定引用类型,那么默认是强引用。
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;
软引用在JVM内存不足时会被回收,用这种特性,可以在一些内存敏感的场景上用软引用。 比如Bitmap对象,可以用软引用
SoftReference<Bitmap> bitmap = new SoftReference<Bitmap>();
弱引用有着比软引用更脆弱的生命周期。即使内存充足,但是只要被GC扫描到就会被回收
WeakReference<String> abcWeakRef = new WeakReference<String>(str);
本文分享自微信公众号 - Android每日一讲(gh_f053f29083b9)
原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。
原始发表时间:2018-04-24
本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。
我来说两句