前言: 看到篇帖子, 国外一个技术面试官在面试senior java developer的时候, 问到一个weak reference相关的问题. 他没有期望有人能够完整解释清楚weak reference是什么, 怎么用, 只是期望有人能够提到这个concept和java的GC相关. 很可惜的是, 20多个拥有5年以上java开发经验的面试者中, 只有两人知道weak reference的存在, 而其中只有一人实际用到过他. 无疑, 在interviewer眼中, 对于weak reference的理解和应用在面试中给了这一个interviewee相当多的加分. 所以, 将我对于这个技术的理解和使用总结在这篇博客里, 希望读者和自己通过读和写这篇帖子, 能够在以后的工作和面试中获益.

在Java里, 当一个对象o被创建时, 它被放在Heap里. 当GC运行的时候, 如果发现没有任何引用指向o, o就会被回收以腾出内存空间. 或者换句话说, 一个对象被回收, 必须满足两个条件: 1)没有任何引用指向它 2)GC被运行.

在现实情况写代码的时候, 我们往往通过把所有指向某个对象的referece置空来保证这个对象在下次GC运行的时候被回收 (可以用java -verbose:gc来观察gc的行为)

Java代码  
  1. Object c = new Car();
  2. c=null;

但是, 手动置空对象对于程序员来说, 是一件繁琐且违背自动回收的理念的.  对于简单的情况, 手动置空是不需要程序员来做的, 因为在java中, 对于简单对象, 当调用它的方法执行完毕后, 指向它的引用会被从stack中popup, 所以他就能在下一次GC执行时被回收了.

但是, 也有特殊例外. 当使用cache的时候, 由于cache的对象正是程序运行需要的, 那么只要程序正在运行, cache中的引用就不会被GC给(或者说, cache中的reference拥有了和主程序一样的life cycle). 那么随着cache中的reference越来越多, GC无法回收的object也越来越多, 无法被自动回收. 当这些object需要被回收时, 回收这些object的任务只有交给程序编写者了. 然而这却违背了GC的本质(自动回收可以回收的objects).

所以, java中引入了weak reference. 相对于前面举例中的strong reference:

Java代码  
  1. Object c = new Car(); //只要c还指向car object, car object就不会被回收

当一个对象仅仅被weak reference指向, 而没有任何其他strong reference指向的时候, 如果GC运行, 那么这个对象就会被回收. weak reference的语法是:

Java代码  
  1. WeakReference<Car> weakCar = new WeakReference(Car)(car);

当要获得weak reference引用的object时, 首先需要判断它是否已经被回收:

Java代码  
  1. weakCar.get();

如果此方法为空, 那么说明weakCar指向的对象已经被回收了.

下面来看一个例子:

Java代码  
  1. package weakreference;
  2. /**
  3. * @author wison
  4. */
  5. public class Car {
  6. private double price;
  7. private String colour;
  8. public Car(double price, String colour){
  9. this.price = price;
  10. this.colour = colour;
  11. }
  12. public double getPrice() {
  13. return price;
  14. }
  15. public void setPrice(double price) {
  16. this.price = price;
  17. }
  18. public String getColour() {
  19. return colour;
  20. }
  21. public void setColour(String colour) {
  22. this.colour = colour;
  23. }
  24. public String toString(){
  25. return colour +"car costs $"+price;
  26. }
  27. }
Java代码  
  1. package weakreference;
  2. import java.lang.ref.WeakReference;
  3. /**
  4. * @author wison
  5. */
  6. public class TestWeakReference {
  7. public static void main(String[] args) {
  8. Car car = new Car(22000,"silver");
  9. WeakReference<Car> weakCar = new WeakReference<Car>(car);
  10. int i=0;
  11. while(true){
  12. if(weakCar.get()!=null){
  13. i++;
  14. System.out.println("Object is alive for "+i+" loops - "+weakCar);
  15. }else{
  16. System.out.println("Object has been collected.");
  17. break;
  18. }
  19. }
  20. }
  21. }

在上例中, 程序运行一段时间后, 程序打印出"Object has been collected." 说明, weak reference指向的对象的被回收了.

值得注意的一点, 即使有car引用指向对象, 且car是一个strong reference, weak reference weakCar指向的对象仍然被回收了. 这是因为java的编译器在发现进入while循环之后, car已经没有被使用了, 所以进行了优化(将其置空?).

当把TestWeakReference.java修改为:

Java代码  
  1. package weakreference;
  2. import java.lang.ref.WeakReference;
  3. /**
  4. * @author wison
  5. */
  6. public class TestWeakReference {
  7. public static void main(String[] args) {
  8. Car car = new Car(22000,"silver");
  9. WeakReference<Car> weakCar = new WeakReference<Car>(car);
  10. int i=0;
  11. while(true){
  12. System.out.println("here is the strong reference 'car' "+car);//use the strong reference in the while loop
  13. if(weakCar.get()!=null){
  14. i++;
  15. System.out.println("Object is alive for "+i+" loops - "+weakCar);
  16. }else{
  17. System.out.println("Object has been collected.");
  18. break;
  19. }
  20. }
  21. }
  22. }

weak reference指向的object就不会被回收了. 因为还有一个strong reference car指向它.

再举个栗子

现在有如下两个类class A class B,在JVM上生成他们两个类的实例分别为 instance a  instance b

有如下表达式:

A a = new A();

B b = new B();

两个强引用对象就生成了,好吧,那么这个时候我做一下修改:

A a = new A();

B b = new B(a);

B的默认构造函数上是需要一个A的实例作为参数的,那么这个时候 A和B就产生了依赖,也可以说a和b产生了依赖,我们再用一个接近内存结构的图来表达:

a是对象A的引用,b是对象B的引用,对象B同时还依赖对象A,那么这个时候我们认为从对象B是可以到达对象A的。

于是我又修改了一下代码

A a = new A();

B b = new B(a);

a = null;

A对象的引用a置空了,a不再指向对象A的地址,我们都知道当一个对象不再被其他对象引用的时候,是会被GC回收的,很显然及时a=null,那么A对象也是不可能被回收的,因为B依然依赖与A,在这个时候,造成了内存泄漏!

那么如何避免上面的例子中内存泄漏呢?

很简单:

A a = new A();

B b = new B(a);

a = null;

b = null;

这个时候B对象再也没有被任何引用,A对象只被B对象引用,尽管这样,GC也是可以同时回收他们俩的,因为他们处于不可到达区域。

弱引用来了!

A a = new A();

WeakReference wr = new WeakReference(a);

//B b = new B(a);

当 a=null ,这个时候A只被弱引用依赖,那么GC会立刻回收A这个对象,这就是弱引用的好处!他可以在你对对象结构和拓扑不是很清晰的情况下,帮助你合理的释放对象,造成不必要的内存泄漏!!

* WeakReference的一个特点是它何时被回收是不可确定的, 因为这是由GC运行的不确定性所确定的. 所以, 一般用weak reference引用的对象是有价值被cache, 而且很容易被重新被构建, 且很消耗内存的对象.

ReferenceQueue

在weak reference指向的对象被回收后, weak reference本身其实也就没有用了. java提供了一个ReferenceQueue来保存这些所指向的对象已经被回收的reference. 用法是在定义WeakReference的时候将一个ReferenceQueue的对象作为参数传入构造函数.

其他类型的references

-SoftReference

soft reference和weak reference一样, 但被GC回收的时候需要多一个条件: 当系统内存不足时(GC是如何判定系统内存不足? 是否有参数可以配置这个threshold?), soft reference指向的object才会被回收. 正因为有这个特性, soft reference比weak reference更加适合做cache objects的reference. 因为它可以尽可能的retain cached objects, 减少重建他们所需的时间和消耗.

WeakReference 学习和使用相关推荐

  1. 记录学习WeakReference发现的问题

    在学习ThreadLocal时发现ThreadLocalMap里的Entry使用到了WeakReference,所以重新学习WeakReference 查看相关博客例如: https://blog.c ...

  2. Android学习笔记之SoftReference软引用,弱引用WeakReference

    SoftReference可以用于bitmap缓存 WeakReference 可以用于handler 非静态内部类和匿名内部类容易造成内存泄漏 private Handler mRemoteHand ...

  3. java中弱引用知识学习WeakHashMap、WeakReference

    先占个坑,今天因为项目需求所以想到了java中的弱引用特性. 首先什么是弱引用? Object a = new Object();Object b = a; 比如上面的代码,b就是a的强引用.当我们对 ...

  4. WPF学习总结1:INotifyPropertyChanged接口的作用

    在代码中经常见到这个接口,它里面有什么?它的作用是什么?它和依赖属性有什么关系? 下面就来总结回答这三个问题. 1.这个INotifyPropertyChanged接口里就一个PropertyChan ...

  5. 多线程学习-基础(十三)(学习参考·网摘) ArrayBlockingQueue源代碼解析(base jdk 1.8)...

    前记: 这个得首先声明一下,以下大部分内容均参考于:https://blog.csdn.net/wx_vampire/article/details/79585794,本随笔只作为学习作用,侵权删! ...

  6. JDK7 源码学习系列——ThreadLocal

    为什么80%的码农都做不了架构师?>>>    学习JDK中的类,首先看下JDK API对此类的描述,描述如下: 该类提供了线程局部 (thread-local) 变量.这些变量不同 ...

  7. C#垃圾回收学习总结

    浅谈C#垃圾回收 http://www.cnblogs.com/cuiyiming/archive/2013/03/26/2981931.html 理解C#垃圾回收机制我们首先说一下CLR(公共语言运 ...

  8. Android之ListView原理学习与优化总结

    在整理前几篇文章的时候有朋友提出写一下ListView的性能优化方面的东西,这个问题也是小马在面试过程中被别人问到的-..今天小马就借此机会来整理下,网上类似的资料蛮多的,倒不如自己写一篇,记录在这个 ...

  9. Android多线程源码学习笔记一:handler、looper、message、messageQueue

    最近在学习Android多线程相关知识的源码,现在把自己的笔记整理一下,写出来加深印象. Android多线程通讯的核心是handler.looper.message.messageQueue,这篇文 ...

最新文章

  1. 产品观,来自微信张小龙的
  2. MATLAB中GUI设计的基本操作
  3. Lesson 5.分类模型决策边界与模型评估指标
  4. nginx源码分析—内存池结构ngx_pool_t及内存管理
  5. Nacos源码更服务列表
  6. php+redis+设置前缀,spring使用Redis自定义前缀后缀名(去掉SimpleKey []+自定义)
  7. 信息学奥赛C++语言:可口可乐
  8. EMNLP2021 论文预讲会,邀你一起共赏自然语言处理学术盛宴(日程全公开)
  9. 通过C#查询SQLServer数据库超时
  10. 如何写数学建模竞赛论文
  11. 数据分析软件SPSS22的授权及汉化
  12. 鸿蒙大陆v2.8正式版,鸿蒙大陆正式版地图下载-鸿蒙大陆下载 V2.22--pc6下载站
  13. 最简单的U盘安装windows系统教程
  14. 终止python程序的死循环_终的解释|终的意思|汉典“终”字的基本解释
  15. Android组件化开发实践和案例分享 | 融合数10个项目模块
  16. 为什么现代物理学离不开量子论和相对论?
  17. 超实用Word小技巧,常用但很少有人记得住
  18. 电脑上怎么同时录制系统和麦克风声音
  19. Linux 重启nginx服务
  20. JavaScript网页购物车

热门文章

  1. java rbac_RBAC基于角色的权限访问控制
  2. 【愚公系列】2023年07月 WPF+上位机+工业互联 002-WPF布局控件
  3. excel表格如何对比两张成绩单的成绩
  4. Windows11,视频教你轻松恢复出厂状态
  5. 微信小程序文章内容展开收缩
  6. c语言conflicting types,gcc编译C程序出现”error conflicting types for function”编译错误的分析解决...
  7. 科奥斯扫地机器人怎么样_科沃斯DL33max扫地机器人怎么样?不是忽悠,真实情况分享!sundhaow...
  8. Android混合开发-(Android与Web的交互)
  9. Python 利用深度学习做文本摘要
  10. listagg乱码问题