WeakReference常常被用来防止内存泄露,最常用的一种情景:
private static class MainHanlder extends Handler {
WeakReference<MainActivity> mActivity = null;
public MainHanlder(MainActivity mainActivity) {
mActivity = new WeakReference<MainActivity>(mainActivity);
}
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case MSG_UPDATE_TV:
mActivity.get().tvHandler.setText("SUCESS");
break;
default:
break;
}
}
}
这样mActivity持有的是MainActivity的弱引用,能够有效地避免内存泄露。 那么为何这样使用,能够避免内存泄露呢? 在发生GC时,WeakReference是怎样被回收的呢?
1、WeakReference在GC中的逻辑
Java的GC分为minor GC和Major GC:
- 发生minor GC时,如果WeakReference引用的对象,位置在新生代,而且只有弱引用,没有其他引用,则会被回收。
- 发生major GC时,如果WeakReference引用的对象,只有弱引用,无论在新生代还是老年代,都会被回收。
这说明,只要堆上的对象仅仅只被弱引用所指向,不管当前内存空间是否足够,下次GC都会回收对象的内存空间。
2、补充Java 内存管理相关
2.1 JVM的结构
2.2 Java内存区域分配
Java的内存区域分为以下五个区域:
程序计数器和寄存器(了解即可)
- 当前线程所执行的指令的地址,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器完成;每个线程都有一个独立的程序计数器;程序计数器只占很小一部分内存空间。
方法区(了解即可)
- 方法区存放的是每个类的信息(类的名称、方法、字段)、静态变量、常量、编译器编译后的代码。
- 很多人习惯将方法区成为“永久代”,是因为HotSpot虚拟机以永久代来实现方法区,不过JDK7以后,HotSpot虚拟机便将运行时常量池,从永久代去除了。
本地方法栈(了解即可)
- 本地方法栈是Java调用通过JNI调用C代码,产生的栈区。
- 本地方法栈油JVM管理,栈内的数据在超出作用域后,会自动被释放掉。
Java栈(重要)
- Java栈中存放的是基本数据类型和对象的引用(划重点,是引用。对象本身是存放在堆中)。
注意:String、Integer、Byte、Short、Long、Character、Boolean这六个属于包装类型,是存放在堆中的。 - 栈的类型类似于水杯,先进后出。而且栈内的数据在超出其作用域后,会被自动释放,不由JVM GC管理。
- 每个线程都会创建一个操作栈,互不干扰,每个栈又包含了若干个栈帧。 每个方法的执行,都会创建一个栈帧,用于存储:局部变量区(方法内的基本类型变量、变量对象指针)、操作数栈区(存放方法执行过程中产生的中间结果)、运行环境区(动态链接、异常捕捉)。 每个方法从调用到执行完成,对应着一个栈帧在虚拟机中入栈-出栈的过程。
Java堆(重要)
堆内存用于存放new出来的对象和数组。
所有的线程共享一个堆,堆是占用内存最多的区域。 堆的内存回收是由GC负责的。
注意,除程序计数器外,其他内存区域当内存不足时,都会报OutOfMemoryError异常。