一,定义

在Java中,引用的定义是:如果reference类型的数据中存储的数值代表的是另一块内存的起始地址,就称这块内存代表着一个引用。后面在JDK1.2开始,引用的概念被扩充,引用被分为强引用(StrongReference)、软引用(SoftReference)弱引用、(WeakReference)、虚引用(PhantomReference)。这四种引用的强度关系:强引用>软引用>弱引用>虚引用。

1,强引用:这种引用就是我们在代码中最常用的,类似于“Object o = new Object()”,这个“o”引用就是强引用,只要强引用还存在,GC永远不会回收掉被引用的对象。 
2,软引用:软引用是用来描述那些有用但非必须的对象,在系统发生内存溢出之前会对这些对象进行回收,也就是说内存不足就会回收这些对象,如果内存足够,即使手动GC也不会回收被软件引用指向的对象。 
3,弱引用:弱引用用来描述非必须对象,强度比软引用更弱,当GC工作时,不管当前内存是否足够,都会回收只被弱引用指向的对象。 
4,虚引用:与其他几种引用不同,虚引用不会影响对象的生存时间,也不能通过虚引用来获得一个对象的实例,它唯一的目的就是对象被回收时能收到一个通知。

二,实例说明

说了几种引用的概念,我们结合一些实例来说明他们各自的特性。

public class MyClass {
public static void main(String[] agrs){
MyClass myClass = new MyClass(20,"Zhang san");
SoftReference<MyClass> softReference = new SoftReference<MyClass>(new MyClass(15,"Li si"));
WeakReference<MyClass> weakReference = new WeakReference<MyClass>(new MyClass(10,"Wang er"));
PhantomReference<MyClass> phantomReference = new PhantomReference<>(new MyClass(5,"Ma zi"), new ReferenceQueue<>()); System.out.println("strong reference myClass = " + myClass);
System.out.println("soft reference get: " + softReference.get());
System.out.println("weak reference get: " + weakReference.get());
System.out.println("phantom reference get: " + phantomReference.get()); } private int age;
private String name; public MyClass(int age, String name){
this.age = age;
this.name = name;
} @Override
public String toString() {
return "name[" + this.name + "] age[" + this.age + "]";
} @Override
protected void finalize() throws Throwable {
super.finalize();
System.out.println("finalize method executed...." + this);
}
}

我们先看上面的例子,一个测试类,main()方法中定义了四个变量,分别是强引用、软引用、弱引用、虚引用,我们看下运行后的打印信息

strong reference myClass = name[Zhang san] age[20]
soft reference get: name[Li si] age[15]
weak reference get: name[Zhang san] age[20]
phantom reference get: null

从打印信息我们可以得知,虚引用虽然指向了一个对象,但是通过该引用得到的对象是null,这就验证了前面我们说的,不能通过虚引用得到对象的实例。 
接下来,我们做点修改,加上显示调用垃圾回收,主要修改如下:

public static void main(String[] agrs){
MyClass myClass = new MyClass(20,"Zhang san");
SoftReference<MyClass> softReference = new SoftReference<MyClass>(new MyClass(15,"Li si"));
WeakReference<MyClass> weakReference = new WeakReference<MyClass>(new MyClass(10,"Wang er"));
PhantomReference<MyClass> phantomReference = new PhantomReference<>(new MyClass(5,"Ma zi"), new ReferenceQueue<>()); System.gc();
System.out.println("strong reference myClass = " + myClass);
System.out.println("soft reference get: " + softReference.get());
System.out.println("weak reference get: " + weakReference.get());
System.out.println("phantom reference get: " + phantomReference.get()); }

运行结果如下:

strong reference myClass = name[Zhang san] age[20]
finalize method executed....name[Ma zi] age[5]
finalize method executed....name[Wang er] age[10]
soft reference get: name[Li si] age[15]
weak reference get: null
phantom reference get: null

从结果中我们可以看到,当显示调用垃圾回收时,弱引用和虚引用对象都被回收了,这说明弱引用和虚引用指向的对象在发生GC时一定会被回收。但强引用和软引用指向的对象并没有被回收。我们接着做些修改,修改如下:

public static void main(String[] agrs){
MyClass myClass = new MyClass(20,"Zhang san");
SoftReference<MyClass> softReference = new SoftReference<MyClass>(new MyClass(15,"Li si"));
WeakReference<MyClass> weakReference = new WeakReference<MyClass>(new MyClass(10,"Wang er"));
PhantomReference<MyClass> phantomReference = new PhantomReference<>(new MyClass(5,"Ma zi"), new ReferenceQueue<>());
myClass = null;
System.gc();
System.out.println("strong reference myClass = " + myClass);
System.out.println("soft reference get: " + softReference.get());
System.out.println("weak reference get: " + weakReference.get());
System.out.println("phantom reference get: " + phantomReference.get()); }

同样看下运行的结果

strong reference myClass = null
finalize method executed....name[Zhang san] age[20]
finalize method executed....name[Ma zi] age[5]
finalize method executed....name[Wang er] age[10]
soft reference get: name[Li si] age[15]
weak reference get: null
phantom reference get: null

可以看出这个时候,强引用指向的对象被回收了,这是因为我们把myClass引用置空了,也就是说name为张三,age为20的对象没被引用了,当发生GC的时候,该对象就会被回收。

三,总结

强引用指向的对象如果被引用,发生GC时是不会被回收的,除非该对象没有被引用;软引用指向的对象在发生GC时不一定会被回收,该对象会被回收的条件是内存不足;弱引用和虚引用指向的对象在发生GC时一定会被回收,此外通过虚引用得不到引用的对象实例。

Java 引用分类:StrongReference、SoftReference、WeakReference、PhantomReference的更多相关文章

  1. Java之引用类型分析(SoftReference/WeakReference/PhantomReference)

    引言: 即使对于Java的很多老鸟来说,如果忽然问他引用的类型,大概率是一脸茫然,不知所措的-.Java中的引用还分类型,神马情况??? 本文将针对这些类型进行分析,帮助您一文知所有类型. Java的 ...

  2. 4种引用与垃圾回收 :StrongReference, SoftReference, WeakReference , PhantomReference

  3. Java引用总结--StrongReference、SoftReference、WeakReference、PhantomReference

    Java引用总结--StrongReference.SoftReference.WeakReference.PhantomReference 1 Java引用介绍 Java从1.2版本开始引入了4种引 ...

  4. Reference SoftReference WeakReference PhantomReference Cleaner 的研究与实践

    最近在看netty的时候看到直接内存的相关概念,为了更详细的了解一下具体原理,搜到了一篇不错的文章 http://lovestblog.cn/blog/2015/05/12/direct-buffer ...

  5. Java核心技术-高级特性(2)- SoftReference, WeakReference and PhantomReference

    Java.lang.ref 是 Java 类库中比较特殊的一个包,它提供了与 Java 垃圾回收器密切相关的引用类.这些引用类对象可以指向其它对象,但它们不同于一般的引用,因为它们的存在并不防碍 Ja ...

  6. Java 引用 WeakReference

    Reference 是一个抽象类,而 SoftReference,WeakReference,PhantomReference 以及 FinalReference 都是继承它的具体类.接下来我们来分别 ...

  7. 你不知道的Java引用

    什么是引用   引用就是保存着一块地址(门牌号)的对象,就像C语言的指针那样,引用可以传递某个数据的地址,如果我们想拿到某一条数据,就要先找到他的地址,然后告诉计算机我去拿这个地址的数据,最后计算机就 ...

  8. Java引用详解-StrongReference SoftReference WeakReference PhantomReference

    1 Java引用介绍 Java从1.2版本开始引入了4种引用,这4种引用的级别由高到低依次为:    强引用  >  软引用  >  弱引用  >  虚引用 ⑴强引用(StrongR ...

  9. 强引用(StrongReference)、弱引用(WeakReference)、软引用(SoftReference)、虚引用(PhantomReference)

    1.强引用(StrongReference) 强引用是使用最普遍的引用.如果一个对象具有强引用,那垃圾回收器绝不会回收它.如下: Object o=new Object(); // 强引用 当内存空间 ...

随机推荐

  1. apache-shiro入门<一>

    Apache Shiro是一个强大而灵活的开源安全框架(本来想传到网盘供大家下载,但是鉴于国内网盘动不动就要关闭清楚用户数据:所以我提供了另一个shiro的中文文档下载链接:http://downlo ...

  2. eclipse调试(debug)的时候,出现Source not found,Edit Source Lookup Path,一闪而过

    问题描述 使用Eclipse调试代码的时候,打了断点,经常出现Source not found,网上找了半天,大部分提示点击Edit Source Lookup Path,添加被调试的工程,然而往往没 ...

  3. Microsoft Visual SourceSafe 6.0 无法关联项目

    最近遇到Microsoft Visual SourceSafe 6.0 安装好以后, 无法关联项目,导致无法进行版本控制,研究以后,发现需要运行一个程序,在安装目录下 ..\Visual Source ...

  4. Android使用C++截屏并显示

    使用android底层自带的截屏源码进行修改后,将截取屏幕的内容再次显示在屏幕上,使屏幕呈现出暂停的效果. android自带的截屏代码在android\JB\frameworks\base\cmds ...

  5. Linux 信号(二)—— signal 函数

    弗洛伊德认为:要解决这些苦恼,当事人就要通过回忆并理解自己早期的童年经历,来获得对潜意识冲突的顿悟.弗洛伊德的疗法被称为“精神分析” (psychoanalysis),在 20 世纪的很长一段时间被心 ...

  6. 在Ubuntu Server 14.04中搭建FTP服务器(VMWare)

    自己搭建ftp服务器,方便主机与虚拟机中的Ubuntu传输文件. 选用的ftp软件为vsftpd. 1.命令行: sudo apt-get install vsftpd 2.安装完配置: vsftpd ...

  7. [No0000AE]在 Visual Studio 中调试 XAML 设计时异常

    在 Visual Studio 中进行 WPF, UWP, Silverlight 开发时,经常会遇到 XAML 设计器由于遭遇异常而无法正常显示设计器视图的情况.很多时候由于最终生成的项目在运行时并 ...

  8. MikroTik RB750r2 操作记录

    1. 客户端的下载 http://www.mikrotik.com/download  下载 winbox 2. Reset重置密码的正确姿势 http://wiki.mikrotik.com/ind ...

  9. STL bind1st bind2nd详解

    STL bind1st bind2nd详解   先不要被吓到,其实这两个配接器很简单.首先,他们都在头文件<functional>中定义.其次,bind就是绑定的意思,而1st就代表fir ...

  10. [LeetCode] Patching Array 补丁数组

    Given a sorted positive integer array nums and an integer n, add/patch elements to the array such th ...