Java四种引用类型
2016-03-10 16:05
435 查看
Java有四种引用类型,strongreference,softreference,weakreference,phantomreference。这四种引用的强度按照上面的顺序依次减弱,下面通过几个例子简单了解一下这四种引用类型。
StrongReference
这个不用多讲了,这是java默认的引用类型,如果不特意使用java.lang.ref下的类,那么程序中的所有引用都是强引用。有强引用存在的对象永远都不会被gc收集,所以在内存不够用时,JVM宁愿抛出OutOfMemoryError这样的错误,也不愿意将强引用对象进行回收。
SoftReference
软引用不会保证对象一定不会被回收,只能最大可能保证。如果内存有剩余,那么软引用对象不会被回收,如果内存不足,那么gc会回收软引用对象。所以这种特性可以用来实现缓存技术。软引用要用java.lang.ref.SoftReference来实现。
![](http://paddy-w.iteye.com/images/spinner.gif)
![](http://paddy-w.iteye.com/images/spinner.gif)
![](http://paddy-w.iteye.com/images/spinner.gif)
也就是说,幽灵引用只能与ReferenceQueue(后面会提到这个类)一起使用。如果一个对象仅有幽灵引用,那么它就像没有任何引用一样,在任何时候都可能被gc回收。幽灵引用主要用来跟踪对象被垃圾回收的活动。
![](http://paddy-w.iteye.com/images/spinner.gif)
ReferenceQueue
如果一个对象只有软引用、弱引用或者幽灵引用,gc在回收对象时,JVM会自动将其引用放入一个ReferenceQueue中。WeakHashMap就是利用了ReferenceQueue来实现清除没有强引用Entry的。将上面的弱引用例子稍微改一下:
按 Ctrl+C 复制代码
public class ReferenceQueueTest{
public static void main(String[] args) {
Object ref = new Object();//ref是Object对象的强引用
System.out.println(ref);
ReferenceQueue<Object> rq=new ReferenceQueue<Object>();//有强引用的ReferenceQueue
WeakReference<Object> sf = new WeakReference<Object>(ref,rq);//构造弱引用时传入ReferenceQueue
System.out.println(sf);
ref = null;//去除对象的强引用,在这里加个断点,进行调试
System.gc();//gc对弱引用对象进行回收
System.out.println(rq.poll());
}
}
![](http://paddy-w.iteye.com/images/spinner.gif)
StrongReference
这个不用多讲了,这是java默认的引用类型,如果不特意使用java.lang.ref下的类,那么程序中的所有引用都是强引用。有强引用存在的对象永远都不会被gc收集,所以在内存不够用时,JVM宁愿抛出OutOfMemoryError这样的错误,也不愿意将强引用对象进行回收。
SoftReference
软引用不会保证对象一定不会被回收,只能最大可能保证。如果内存有剩余,那么软引用对象不会被回收,如果内存不足,那么gc会回收软引用对象。所以这种特性可以用来实现缓存技术。软引用要用java.lang.ref.SoftReference来实现。
1 public class SoftTest{ 2 public static void main(String[] args) { 3 Object ref = new Object();//ref是Object对象的强引用 4 5 //将一个软引用指向对象,此时Object对象有两个引用 6 SoftReference<Object> sf = new SoftReference<Object>(ref); 7 8 ref = null;//去除对象的强引用 9 System.gc();//gc只有在内存不足是才会回收软引用对象 10 } 11 }
![](http://paddy-w.iteye.com/images/spinner.gif)
WeakReference 除了通过java.lang.ref.WeakReference来使用弱引用,WeakHashMap同样也利用了弱引用。 和软引用不同的是,弱引用一定会被gc回收,不管内存是否不足。
1 public class WeakTest{ 2 public static void main(String[] args) { 3 Object ref = new Object();//ref是Object对象的强引用 4 5 //将一个弱引用指向对象,此时Object对象有两个引用 6 WeakReference<Object> wf = new WeakReference<Object>(ref); 7 8 ref = null;//去除对象的强引用 9 System.gc();//gc对弱引用对象进行回收 10 } 11 }
![](http://paddy-w.iteye.com/images/spinner.gif)
PhantomReference 幽灵引用,也叫虚引用。java.lang.ref.PhantomReference类中只有一个方法get(),而且几乎没有实现,只是返回null。而且这个类只有一个构造器(软引用和弱引用均有两个构造器):
1 public PhantomReference(T referent, ReferenceQueue<? super T> q) { 2 super(referent, q); 3 }
![](http://paddy-w.iteye.com/images/spinner.gif)
也就是说,幽灵引用只能与ReferenceQueue(后面会提到这个类)一起使用。如果一个对象仅有幽灵引用,那么它就像没有任何引用一样,在任何时候都可能被gc回收。幽灵引用主要用来跟踪对象被垃圾回收的活动。
1 public class PhantomTest{ 2 public static void main(String[] args) { 3 Object ref = new Object();//ref是Object对象的强引用 4 5 //将一个幽灵引用指向对象,PhantomReference必须与ReferenceQueue一同使用 6 PhantomReference<Object> pf = new PhantomReference<Object>(ref, new ReferenceQueue<Object>()); 7 8 System.out.println(pf.get()); 9 } 10 }
![](http://paddy-w.iteye.com/images/spinner.gif)
ReferenceQueue
如果一个对象只有软引用、弱引用或者幽灵引用,gc在回收对象时,JVM会自动将其引用放入一个ReferenceQueue中。WeakHashMap就是利用了ReferenceQueue来实现清除没有强引用Entry的。将上面的弱引用例子稍微改一下:
按 Ctrl+C 复制代码
public class ReferenceQueueTest{
public static void main(String[] args) {
Object ref = new Object();//ref是Object对象的强引用
System.out.println(ref);
ReferenceQueue<Object> rq=new ReferenceQueue<Object>();//有强引用的ReferenceQueue
WeakReference<Object> sf = new WeakReference<Object>(ref,rq);//构造弱引用时传入ReferenceQueue
System.out.println(sf);
ref = null;//去除对象的强引用,在这里加个断点,进行调试
System.gc();//gc对弱引用对象进行回收
System.out.println(rq.poll());
}
}
![](http://paddy-w.iteye.com/images/spinner.gif)
注意,运行这个程序需要用debug模式进行调试,在上面说明的地方加个断点。如果直接运行,结果很可能只是个null,用debug调试的话会看到输出的是弱引用的地址。程序输出的sf结果和rq.poll()结果相同。
相关文章推荐
- java的concurrent用法详解
- 通过分析 JDK 源代码研究 TreeMap 红黑树算法实现
- Spring , duboo 注意事项
- java 交集性能测试
- Spring @Scheduled 任务 参数动态配置
- Java线程之FutureTask
- Java中如何利用Selenium获取元素分析网页内容
- SQLite在Java中的使用
- 深入理解Java虚拟机——程序编译与代码优化 (转)
- 利用btrace在线监控java程序状态
- 在 Eclipse 下利用 gradle 构建系统
- Spring 4.x实现Restful web service
- java spark WordCount
- 动态代理 jdk as cglib asm
- Javac编译器源代码分析
- java实现动态切换上网IP (ADSL拨号上网)
- JAVA 中BIO,NIO,AIO的理解
- Java BIO、NIO、AIO
- Java数据类型和MySql数据类型对应表
- java jvm 参数 -Xms -Xmx -Xmn -Xss 调优总结