近期研究Java Cache实现,发现使用到了软引用(SoftReference),不太理解,查阅了JDK文档、代码以及几篇文章。做个小结,如有错误,欢迎指正。

之所以想学习一下Java的几种引用类型,原因有两个:

  • 理解Java Cache实现、学习Java引用与Java垃圾回收机制的关系

    内存资源是有限的,须要合理的利用。Cache不是只HashMap那么简单,Java引用与Java垃圾回收机制也有很紧密的关系。
  • 避免对Java引用的错误使用

    某个同事把5000+交易数据放到一个HashMap里面,用一个Spring Singleton Bean的全局属性指向该HashMap。大量运用这样的技术,非常快就报out of memory。再大的内存也架不住对内存的错误使用。理解原理有助于我们尽量少犯或不犯低级错误。

Java引用与Java垃圾回收机制的关系

当Java虚拟机(JVM)认为内存不够用的时候,会触发垃圾回收操作(GC),清除没用的对象,释放内存。但是怎样推断一个对象是否是垃圾呢?当中的一个方法是计算指向该对象的引用数量,假设引用数量为0,那么该对象就为垃圾(Thread对象是例外),否则还实用处,不能被回收。但是假设把引用数为0的对象都回收了,还是不能满足内存需求怎么办?Java把引用分为4种类型,垃圾回收器会尝试回收仅仅有弱引用的对象。

依照一个对象的引用可达(Reachable)强度,由强到弱分为5类,例如以下:

  • 强可达(Strong Reachable)

    在一个线程内,无需引用直接可达,新创建的对象是强可达的。
  • 软可达(Soft Reachable)

    不是强可达的,可是通过一个软引用(SoftReference)可达。
  • 弱可达(Soft Reachable)

    既不是强可达也不是软可达,可是通过一个弱引用(WeakReference)可达。
  • 虚可达(Phantom Reachable)

    既不是强可达,不是软可达,也不是弱可达,可是通过一个虚引用(PhantomReference)可达。
  • 不可达(Unreachable)

    没有不论什么引用指向对象。

比較好、easy理解的是Java垃圾回收器会优先清理可达强度低的对象。另外有两个重要的点:

  • 强可达的一定不会被清理
  • JVM保证抛出out of memory之前,清理全部的软引用对象

4种Java引用

在实现一个缓存系统的时候,假设所有使用强引用,那么你须要自己去手动的把某些引用clear掉(引用置位null),否则迟早会抛出out of memory错误。缓存系统引入弱引用或者软引用的唯一原因是,把引用clear的事情交由Java垃圾回收器来处理,cache程序自己置身事外

几种弱引用的使用方式很相近。以下分别介绍4种引用类型。

强引用(StrongReference)

我们平时申明变量使用的就是强引用,普通系统99%以上都是强引用。比方,String s = "Hello World"

弱引用(WeakReference)

垃圾回收器某个时刻决定回收软可达的对象的时候,会清理软引用,并可选的把引用存放到一个引用队列(ReferenceQueue)。

软引用(SoftReference)

类似弱引用,仅仅只是Java虚拟机会尽量让软引用的存活时间长一些,迫不得已才清理。

虚引用(PhantomReference)

仅用来处理资源的清理问题,比Object里面的finalize机制更灵活。get方法返回的永远是null,Java虚拟机不负责清理虚引用,可是它会把虚引用放到引用队列里面。

两个測试样例

使用HashMap,会报out of memory错误。

public static void main(String[] args) {
Map<String, String> list = new HashMap<String, String>();
long i = 1;
while (i < 100000000L) {
list.put(
String.valueOf(i),
"JDJJDJJJJJJJJJJ%%%%%%%%JJJJJJJJJJJJJJJKKKKKKKKKKKKKKKKKJJJJJJ"
+ "JJJKKKKKHDDDJDJDJDJDJDJDJDJJDJDJDJDJDJDJJDJDJDJDJJDJDJJJJJJJJJ"
+ "JJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJ"
+ "JJJJJJJJJJJJJJJJJJJJJJJJJJJJ"); // 測试第一个是否依旧存活
if (i % 100000 == 0) {
System.out.println(list.get(String.valueOf(1)));
} i++;
}
}

使用WeakHashMap,不会报out of memory错误。

public static void main(String[] args) {
Map<String, String> list = new WeakHashMap<String, String>();
long i = 1;
while (i < 100000000L) {
list.put(
String.valueOf(i),
"JDJJDJJJJJJJJJJ%%%%%%%%JJJJJJJJJJJJJJJKKKKKKKKKKKKKKKKKJJJJJJ"
+ "JJJKKKKKHDDDJDJDJDJDJDJDJDJJDJDJDJDJDJDJJDJDJDJDJJDJDJJJJJJJJJ"
+ "JJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJJ"
+ "JJJJJJJJJJJJJJJJJJJJJJJJJJJJ"); // 測试第一个是否依旧存活
if (i % 100000 == 0) {
System.out.println(list.get(String.valueOf(1)));
} i++;
}
}

小结

Java语言里面数组(Array)、列表(List)、Map等容器,对里面的每个对象都有一个引用,大数据的情况下要小心内存泄露。弱引用仅仅适合cache等特殊场景,对于那些一定不能Java让垃圾回收器回收的对象,要使用强引用。

參考连接

4种Java引用浅解的更多相关文章

  1. 强引用,弱引用,4种Java引用浅解(涉及jvm垃圾回收)

    http://www.jb51.net/article/49085.htm http://www.jb51.net/article/49085.htm

  2. 9种Java单例模式详解(推荐)

    单例模式的特点 一个类只允许产生一个实例化对象. 单例类构造方法私有化,不允许外部创建对象. 单例类向外提供静态方法,调用方法返回内部创建的实例化对象.  懒汉式(线程不安全) 其主要表现在单例类在外 ...

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

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

  4. 9种Java单例模式详解

    单例模式的特点 一个类只允许产生一个实例化对象. 单例类构造方法私有化,不允许外部创建对象. 单例类向外提供静态方法,调用方法返回内部创建的实例化对象. 懒汉式(线程不安全) 其主要表现在单例类在外部 ...

  5. Java引用详解

    区分几个概念 ,类 .对象 .对象的引用 A  a1  = new A(); A a2 = a1; a1-------------------->对象内存<---------------- ...

  6. Java四大引用详解:强引用、软引用、弱引用、虚引用

    面试官考察Java引用会问到强引用.弱引用.软引用.虚引用,具体有什么区别?本篇单独来详解 @mikechen Java引用 从JDK 1.2版本开始,对象的引用被划分为4种级别,从而使程序能更加灵活 ...

  7. java基础-四种方法引用

    实例 直接三角形,通过两边算第三边,目的是为了如何使用这几种方法引用.代码中多有些不合适,尽情原谅. 静态方法引用 接口的参数列表与类中的具体实现方法的参数列表一样,返回值一致. 调用 //静态引用 ...

  8. 浅谈Java引用和Threadlocal的那些事

      这篇文章主要介绍了Java引用和Threadlocal的那些事,小编觉得挺不错的,现在分享给大家,也给大家做个参考.一起跟随小编过来看看吧 1 背景 某一天在某一个群里面的某个群友突然提出了一个问 ...

  9. java 泛型详解-绝对是对泛型方法讲解最详细的,没有之一

    对java的泛型特性的了解仅限于表面的浅浅一层,直到在学习设计模式时发现有不了解的用法,才想起详细的记录一下. 本文参考java 泛型详解.Java中的泛型方法. java泛型详解 1. 概述 泛型在 ...

随机推荐

  1. Linux学习之挂载

    linux的系统组织方式是——整个系统从根开始,按树形目录依次向下逐渐扩大,分类存放不同用途的文件,/读作“斜线”,英文slash:当其写作一个路径时,第一个/表示根,即root,其他的/表示路径分割 ...

  2. Aliyun EMR 集群重启

    1.如果Core节点有Down掉,ActiveNodes少于Core节点数. 处理: a.登陆到Master节点,到目录 /opt/apps/hadoop-2.7.2/sbin b.执行 ./stop ...

  3. java 线程学习

    转载:详见处http://lavasoft.blog.51cto.com/62575/27069   Java多线程编程总结   下面是Java线程系列博文的一个编目:   Java线程:概念与原理 ...

  4. [转]关于HTTP服务器每个客户端2个连接的限制

    这两天猫在家里搞一个多线程的断点续传得C#程序,发现同时只能开2个线程下载,其他的线程一律要等待,这样就导致下载大文件时其他线程经常超时,郁闷好久.今天回公司无意中发现了一个帖子,终于真相大白了, 现 ...

  5. 让Java的反射跑快点

    由于反射涉及动态解析的类型,某些Java虚拟机的优化不能被执行,所以导致了一定的性能的问题,特别是在JDK6以前特别严重,有时甚至达到数百倍,但是在JDK6以后,据说性能差别就不是哪么大了,JDK对此 ...

  6. javascript简单对象创建

    由于javascript中定义了一个函数就相当于定义了一个类,我们当然可以创建一个这个类的对象. <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 ...

  7. ios 中生成二维码和相册中识别二维码

    iOS 使用CIDetector扫描相册二维码.原生扫描 原生扫描 iOS7之后,AVFoundation让我们终于可以使用原生扫描进行扫码了(二维码与条码皆可)AVFoundation可以让我们从设 ...

  8. iOS加载HTML, CSS代码

    NSString *strHTML = @"<div style=\"text-align:center;\"><img src=\"/Upl ...

  9. linux下coredump的产生及调试方法

    什么是coredump 通常情况下coredmp包括了程序执行时的内存,寄存器状态,堆栈指针,内存管理信息等.能够理解为把程序工作的当前状态存储成一个文件.很多程序和操作系统出错时会自己主动生成一个c ...

  10. MOSS程序中如何发Mail?

    我们使用.NET类库中的API发Mail的时候,我们要配置他的SMTP Server等,但是在Sharepoint里,已经提供了相关的封装的方法: SPUtility.SendEmail(SPWeb, ...