java的Reference学习

java中Reference学习

谈到Reference想到了什么

Reference提供了一种与jvm gc交互的一种方式,提到Reference,脑中应该浮现一些名词,gc、ReferenceQueue、SoftReference、WeakReference、PhantomReference、FinalReference以及最常见的强引用。我认为当一个小白开始学习Reference的时候应该想一个问题,强引用不够好么,为什么我还需要弱引用、软引用等等这些东西。

为什么我们需要它们

应用程序经常会遇到这样一种情况,有些类天然不适合被拓展,这些类可能被标记为了final,或者可能情况更加复杂些,它是被一个工厂方法返回,你压根就不知道它具体的实现类,你仅仅知道它是某个接口,然而你想要拓展他们,因此你可能使用了一个map,key是他们的示例,value是你想使用的一些拓展属性,比如一个名字,Ethan Nicholas‘s Blog举了个例子:

Widget类就不能被拓展,而我们希望给这些对象一些序列号widgetSerialNumber,因此我们可能会这样:

serialNumberMap.put(widget, widgetSerialNumber);

看起来一些很美好,但是这个map拥有着widget的强引用,当这个类不再被需要的时候,我们可能需要手动删除,否则它不可能被gc。看起来问题回到了没有gc的时代(C++手动管理内存),我们不得不时刻考虑着这块内存的所有权,当应该释放的时候把它释放掉,这太不java了。

因此有些美好的特性就出来帮你解决这个问题,

WeakReference weakWidget = new WeakReference(widget);

通过一层弱引用封装,jvm在做可达性分析的时候就可以再正确的时机将他们清理调。因此我们再代码中就可能会偶然发现weakWidget.get()返回null,说明已经被gc回收了。

Reference queues

没个Reference对象在构建的时候可以传递一个ReferenceQueue对象示例,这个对象是我们面对gc时候的一把钥匙,从名字就可以看出它就是一个队列(翻看源代码我觉得叫栈更合适,因为总是在head插入,head出删除),准确的说其实就是个链表,链表的元素就是Reference对象,可以执行一些基本的入队、删除操作,注意到这个remove是阻塞的,而poll并不阻塞而是立即返回null假如队列没有元素。

为什么说它是一把钥匙,因为一般在使用的时候我们会构造一个ReferenceQueue示例对象,然后在后面构造弱引用等类型时候传入,这样当他们被gc回收的时候,remove方法就会返回被gc的示例,我们就可以做一些清理工作,这个模型非常像事件模型,当然对于weakReference而言,入队并不一定会被gc,因为有finalize方法存在,可能会复活一些对象,但是对于PhantomReference而言,入队了这块内存一定被gc了,因此PhantomReference可以作为一个gc时间的listener(有get到一个小trick),但是使用起来其实并不是很方便。

引用的强度

很明显,从名字就可以看出来,引用强度从大到小: strong > soft > weak > phantom。

其中soft和weak的引用其实很像,区别在于weak引用一旦发生gc就会将它回收,而soft引用会尽可能撑到没有内存可用的时候,被回收。

Phantom引用更以上都不太一样,它几乎不能够持有一个对象,它get()总是返回null,唯一的作用就是从ReferenceQueue中取出来使用,当取出来时候我们就知道它已经被回收了,我们可以很放心做一些该有的回收动作。

当我在学习的时候,我一直想不明白,他们一被gc就回收,这用起来多不放心,但是当我接触到几个demo代码的时候我就明白了,他们从来不是在单兵作战,基本上他们的使用都会和强引用配合。

因此对象可达性分析就很重要了,实际上以上的情况准确说是weak可达性或者soft可达性或者phantom可达性。

下面引用一下java学习教程之Reference详解中的图示:

那么 垃圾回收时会依据两个原则来判断对象的可达性:

  1. 单一路径中,以最弱的引用为准
  2. 多路径中,以最强的引用为准。

有这两句话所有的解答就出来了。

实际的一些例子后面补,先挖个坑。

原文地址:https://www.cnblogs.com/snailLx/p/9165265.html

时间: 2024-10-03 08:24:33

java的Reference学习的相关文章

Java 线程池学习

Reference: <创建Java线程池>[1],<Java线程:新特征-线程池>[2], <Java线程池学习>[3],<线程池ThreadPoolExecutor使用简介>[4],<Java5中的线程池实例讲解>[5],<ThreadPoolExecutor使用和思考>[6] [1]中博主自己通过ThreadGroup实现一个线程池(挺方便理解的),使用的是jdk1.4版本,Jdk1.5版本以上提供了现成的线程池. [2]中介绍

Java泛型通配符学习 —— Java Generic&#39;s Wildcards

Java Generic's wildcards is a mechanism in Java Generics aimed at making it possible to cast a collection of a certain class, e.g A, to a collection of a subclass or superclass of A. This text explains how. 理解:Java的泛型通配符机制旨在实现集合的类型转换.例如集合A,转换为A的子类集合或

java/android 设计模式学习笔记(14)---外观模式

这篇博客来介绍外观模式(Facade Pattern),外观模式也称为门面模式,它在开发过程中运用频率非常高,尤其是第三方 SDK 基本很大概率都会使用外观模式.通过一个外观类使得整个子系统只有一个统一的高层的接口,这样能够降低用户的使用成本,也对用户屏蔽了很多实现细节.当然,在我们的开发过程中,外观模式也是我们封装 API 的常用手段,例如网络模块.ImageLoader 模块等.其实我们在开发过程中可能已经使用过很多次外观模式,只是没有从理论层面去了解它. 转载请注明出处:http://bl

java/android 设计模式学习笔记(10)---建造者模式

这篇博客我们来介绍一下建造者模式(Builder Pattern),建造者模式又被称为生成器模式,是创造性模式之一,与工厂方法模式和抽象工厂模式不同,后两者的目的是为了实现多态性,而 Builder 模式的目的则是为了将对象的构建与展示分离.Builder 模式是一步一步创建一个复杂对象的创建型模式,它允许用户在不知道内部构建细节的情况下,可以更精细地控制对象的构造流程.一个复杂的对象有大量的组成部分,比如汽车它有车轮.方向盘.发动机.以及各种各样的小零件,要将这些部件装配成一辆汽车,这个装配过

Java程序猿学习当中各个阶段的建议

回答阿里社招面试如何准备,顺便谈谈对于Java程序猿学习当中各个阶段的建议 引言 其实本来真的没打算写这篇文章,主要是LZ得记忆力不是很好,不像一些记忆力强的人,面试完以后,几乎能把自己和面试官的对话都给记下来.LZ自己当初面试完以后,除了记住一些聊过的知识点以外,具体的内容基本上忘得一干二净,所以写这篇文章其实是很有难度的. 但是,最近问LZ的人实在是太多了,为了避免重复回答,给自己省点力气,干脆就在这里统一回复了. 其实之前LZ写过一篇文章,但是那篇文章更多的是在讨论“面试前该不该刷题”这个

java/android 设计模式学习笔记(一)---单例模式

前段时间公司一些同事在讨论单例模式(我是最渣的一个,都插不上嘴 T__T ),这个模式使用的频率很高,也可能是很多人最熟悉的设计模式,当然单例模式也算是最简单的设计模式之一吧,简单归简单,但是在实际使用的时候也会有一些坑. PS:对技术感兴趣的同鞋加群544645972一起交流 设计模式总目录 java/android 设计模式学习笔记目录 特点 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 单例模式的使用很广泛,比如:线程池(threadpool).缓存(cache).对

Java Level 2 学习的八大名著

Java Level 2 学习的八大名著 前段时间有几天难得的假期,于是把自己认为Java技术栈中的精华总结了一下,但是一直没有时间写下来,今天终于得空希望本文可以对大家有所启发.通过多个实际项目的沉淀和一些Java开源项目的源代码研究,个人认为Java语言中的精华技术主要包括如下三个方面,其实严格来说只有第一点说的是纯Java中的东西,提到二,三两点是因为这两种技术在Java技术栈中应用的比较广泛,而且对于Java技术栈的意义比较大. 1. 内存模型,并发基础设施,IOs 2. 开源MQ 3.

一位资深程序员大牛给予Java初学者的学习路线建议(转)

java学习这一部分其实也算是今天的重点,这一部分用来回答很多群里的朋友所问过的问题,那就是我你是如何学习Java的,能不能给点建议?今天我是打算来点干货,因此咱们就不说一些学习方法和技巧了,直接来谈每个阶段要学习的内容甚至是一些书籍.这一部分的内容,同样适用于一些希望转行到Java的同学. 在大家看之前,我要先声明两点.1.由于我本人是Java后端开发出身,因此所推荐的学习内容是Java Web和Java后端开发的路线,非Java Web和Java后端开发的同学请适当参考其学习思想即可,切勿照

java/android 设计模式学习笔记(7)---装饰者模式

这篇将会介绍装饰者模式(Decorator Pattern),装饰者模式也称为包装模式(Wrapper Pattern),结构型模式之一,其使用一种对客户端透明的方式来动态的扩展对象的功能,同时它也是继承关系的一种替代方案之一,但比继承更加灵活.在现实生活中也可以看到很多装饰者模式的例子,或者可以大胆的说装饰者模式无处不在,就拿一件东西来说,可以给它披上无数层不一样的外壳,但是这件东西还是这件东西,外壳不过是用来扩展这个东西的功能而已,这就是装饰者模式,装饰者的这个角色也许各不相同但是被装饰的对