在Java里面,是不需要太过于关乎垃圾回收,但是这并不意味着开发者可以不了解垃圾回收的机制,况且在java中内存泄露也是家常便饭的事情。因此了解垃圾回收的相关知识就显得很重要了。

引用,在垃圾回收中是一个很关键的概念,它关系到如何辨别这个对象是否被回收,什么时机回收。

引用的类型

在Java中引用的类型可以分为四个类型,依次是:

  • 强引用:在任何时间JVM都不会进行回收
  • 软引用:在内存不够的时候,JVM会进行回收
  • 弱引用:只要进行垃圾回收,就会触发回收
  • 虚引用:不知道啥时候就被回收了,可以理解为没引用一个样

因此,按照JVM对他们回收的几率从小到大依次为:

强引用<软引用<弱引用<虚引用

也就是说JVM对强引用的回收能力最小,对虚引用的回收能力最大。

引用分类的作用

一般我们编写的代码都是强引用的,比如:

Person p = new Person();
Person p1 = p;

pp1都指向了创建的Person对象,他们都是强引用的。如果想要回收这个对象,只有p1p都指向null后,才可以。

那么,有一些场景下往往引用清除的不及时,就会造成内存泄露,一些对象无法回收;无法回收的对象如果积累很多,就会造成OUT OF MEMORY——OOM.

举个例子,在很多的代码里面都喜欢用Map作为内存缓存的容器,如果你写出了这样的代码:

Map<String,Object> map = new HashMap<String,String>();
while(true){
Object value = new XXX();
map.add(key,value);
value = null;
}

虽然说,后面的value被设置成Null,但是map里面却仍然保留了对象的引用,因此这个对象实际上是无法被回收的。

做个测试:

public class WeakTest {
static final int MB = 1024 * 512; static String createLongString(int length) {
StringBuilder sb = new StringBuilder(length);
for (int i = 0; i < length; i++)
sb.append('a');
sb.append(System.nanoTime());
return sb.toString();
} public static void main(String[] args) {
Map<Integer,String> substrings = new HashMap();//强引用的Map
for(int i=0; i< 1000000; i++){
String longStr = createLongString(MB);
substrings.put(i,longStr);
// longStr = null;
// substrings.remove(i);
System.out.println(i);
}
}
}

如果注释的两句话不被打开,那么很快就会内存溢出。除非你两边都去解除应用,可想而知,程序员做这种工作实在是太痛苦了。

不要担心,这个时候就可以应用到上面的不同类型的引用知识了

在Java里面,JDK为我们提供了一个弱引用的集合,WeakHashMap。它会在垃圾回收的时候尝试回收集合里面的对象。当然根据垃圾回收的时机,也可以选择软引用的集合。

public static void main(String[] args) {
Map<Integer,String> substrings = new WeakHashMap();//弱引用的Map
for(int i=0; i< 1000000; i++){
String longStr = createLongString(MB);
substrings.put(i,longStr);
System.out.println(i);
}
}

这样就不担心内存溢出了。

场景设想

比如,你的系统需要引用大量的资源相关的缓存,但是还没有引入redis等缓存系统,那么就可以使用这种方式。

虚引用

虚引用的使用场景就比较鸡肋了,我也想不出什么时候会使用它。但是它跟其他的引用都有一种场景,就是在垃圾回收的时候,把引用放在回收队列里面,针对这个队列可以做一些操作。这种方式比finalize()要文档的多..

public class PhantomTest {
public static boolean isRun = true; public static void main(String[] args) throws Exception {
String abc = new String("abc");
System.out.println(abc.getClass() + "@" + abc.hashCode());
final ReferenceQueue referenceQueue = new ReferenceQueue<String>();
new Thread() {
public void run() {
while (isRun) {
Object o = referenceQueue.poll();
if (o != null) {
try {
Field rereferent = Reference.class.getDeclaredField("referent");
rereferent.setAccessible(true);
Object result = rereferent.get(o);
System.out.println("gc will collect:"+ result.getClass() + "@"+ result.hashCode());
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
}.start();
PhantomReference<String> abcWeakRef = new PhantomReference<String>(abc,referenceQueue);
abc = null;
Thread.currentThread().sleep(3000);
System.gc();
Thread.currentThread().sleep(3000);
isRun = false;
} }

首先需要创建一个引用队列:

final ReferenceQueue referenceQueue = new ReferenceQueue<String>();

创建虚引用,并关联到引用队列

PhantomReference<String> abcWeakRef = new PhantomReference<String>(abc,referenceQueue);

等引用被回收的时候,就会在Object o = referenceQueue.poll();取到对象引用了。

虽然一般不会有这种底层的使用场景,但是了解一点总归是好的。

Java程序员的日常—— 垃圾回收中引用类型的作用的更多相关文章

  1. Java程序员的日常——经验贴(纯干货)

    工作当中遇到的事情比较杂,因此涉及的知识点也很多.这里暂且记录一下,今天遇到的知识点,纯干货~ 关于文件的解压和压缩 如果你的系统不支持tar -z命令 如果是古老的Unix系统,可能并不认识tar ...

  2. Java程序员的日常—— 《编程思想》关于类的使用常识

    Java虽然利用JVM,让程序员可以放心大胆的使用,可是仍然会出现内存泄露等问题.世上没有绝对的银弹,因此也不能完全把所有的任务都交给JVM,了解Java中的初始化与垃圾回收还是必不可少的知识. 关于 ...

  3. Java程序员的日常—— 基于类的策略模式、List<?>与List、泛型编译警告、同比和环比

    早晨起得太早,昨晚睡得太晚,一天都迷迷糊糊的.中午虽然睡了半个小时,可是依然没有缓过来.整个下午都在混沌中....不过今天下载了一款手游--<剑侠情缘>,感觉不错,喜欢这种类型的游戏. 今 ...

  4. Java程序员的日常—— Properties文件的读写

    在日常的Java程序开发中,Properties文件的读写是很常用的.经常有开发系统通过properties文件来当做配置文件,方便用户对系统参数进行调整. 那么本片就来简单的介绍下,如何使用Prop ...

  5. Java程序员的日常 —— 多进程开发

    最近再弄进程管理相关的工作,因此必要的就涉及到各种系统下关于进程的管理. 这里简单的介绍下: 如何在Java中执行命令 在windows下肯定是dos命令了,而在linux则为shell命令.执行的方 ...

  6. Java程序员的日常—— Arrays工具类的使用

    这个类在日常的开发中,还是非常常用的.今天就总结一下Arrays工具类的常用方法.最常用的就是asList,sort,toStream,equals,copyOf了.另外可以深入学习下Arrays的排 ...

  7. Java程序员的日常 —— static的用法讲解实践

    之前文章说过Java中static的作用,有朋友想看个例子.于是便抽空写了个小栗子 代码 package xing.test.thinking.chap5; class A{ public A() { ...

  8. Java程序员的日常 —— Java类加载中的顺序

    之前说过Java中类的加载顺序,这次看完继承部分,就结合继承再来说说类的加载顺序. 继承的加载顺序 由于static块会在首次加载类的时候执行,因此下面的例子就是用static块来测试类的加载顺序. ...

  9. Java程序员的日常——存储过程知识普及

    存储过程是保存可以接受或返回用户提供参数的SQL语句集合.在日常的使用中,经常会遇到复杂的业务逻辑和对数据库的操作,使用存储过程可以进行封装.可以在数据库中定义子程序,然后把子程序存储在数据库服务器, ...

随机推荐

  1. javaScript获取指定的cookie值

    1.获取cookie的函数 function getCookie(cookieName) { var strCookie = document.cookie; var arrCookie = strC ...

  2. XlFileFormat

    -----转载:http://hi.baidu.com/liu_haitao/item/900ddb38979188c22f8ec26e 18 XlFileFormat.xlAddIn Microso ...

  3. (转)DEDECMS模板原理、模板标签学习 - .Little Hann

    本文,小瀚想和大家一起来学习一下DEDECMS中目前所使用的模板技术的原理: 什么是编译式模板.解释式模板,它们的区别是什么? 模板标签有哪些种类,它们的区别是什么,都应用在哪些场景? 学习模板的机制 ...

  4. c-函数指针(求奇数偶数的和)

    #include <stdio.h> /* 编写一个函数,输入 n 为偶数时,调用函数求 1/2+1/4+...+1/n,当输入 n 为奇数时,调用函数1/1+1/3+...+1/n(利用 ...

  5. jquery val() and text().

    .val() works on input elements (or any element with a value attribute?) and .text() will not work on ...

  6. C语言的运行机制

    目的:通过分析c语言转换成汇编代码后的执行过程对汇编语言和X86构架有一个初步认识 实验代码 1 #include <stdio.h> 2 3 int g(int x) 4 { 5 ret ...

  7. MySQL 删除数据库

    MySQL 删除数据库 使用 mysqladmin 删除数据库 使用普通用户登陆mysql服务器,你可能需要特定的权限来创建或者删除 MySQL 数据库. 所以我们这边使用root用户登录,root用 ...

  8. 关于USACO Training

    做了这么久的题目,突然发现最经典的 USACO Training 还没有做过?加速水一遍吧!我会把题解放在上面的.

  9. WebService cxf 接口中获得拦截器参数

    1. 拦截器中put属性 Message message = PhaseInterceptorChain.getCurrentMessage(); message.put("AuthCode ...

  10. centos7 服务器安装nginx,mysql,php

    一.概述 项目的需要,今天在虚拟机上基于Centos安装配置了服务器运行环境,web服务用 nginx,数据库存储在mysql,动态脚本语言是php. 二.步骤 首页保证Centos7已经安装完毕,正 ...