JAVA 四大引用

传说中的区别:

    Java 4种引用的级别由高到低依次为:

                 强引用 > 软引用 > 弱引用 > 虚引用

个人从垃圾回收的速度来评论:  软引用(内存不足) > 强引用(GC算法决定 | 内存不足)   > 弱引用 ≈ 虚引用

1. 强引用 :生命周期结束后,等着下一轮GC来回收

生命周期:正常用对象,不用借助什么weakReference等,对象生命周期方法结束、while每轮循环结束等 (生命周期结束后,等着下一轮GC来回收)

Object  obj   =   new Object();

--------------          ------------------

    强引用              被引用对象

 代表的是 前部分的 “声明变量”:List list  、 Object o都是强引用;这个引用保存在栈中,而真正的引用内容(Object)保存在堆中;当这个方法运行完成后就会退出方法栈,则引用内容的引用不存在(可能被隐式赋值null,指针修改 | while循环内每轮循环结束也是被消除引用),这个Object被引用对象会被回收。

2.软引用:特殊修饰,从缓存中查看对象是否已被回收,则重新创建

内存不足时触发GC?正常内存不足也会触发GC,那和强引用(正常的生命周期结束)也类似啊。。

>>>>> 软引用可用来实现内存敏感的高速缓存

生命周期:内存不足时回收,剔除缓冲中对象

String str=new String("abc");                                     // 强引用

SoftReference softRef=new SoftReference(str);     // 软引用

当内存不足时,等价于:

if(JVM.内存不足()) {

str = null;  // 转换为软引用

System.gc(); // 垃圾回收器进行回收

}

-------------------------------------------------------------------------------------------------------------------------------

如果将浏览过的网页存储到内存中会造成内存的大量浪费,甚至会造成内存溢出

这时候就可以使用软引用

Browser prev = new Browser();               // 获取页面进行浏览

SoftReference sr = new SoftReference(prev); // 浏览完毕后置为软引用

if(sr.get() != null){

rev = (Browser) sr.get();           // 还没有被回收器回收,直接获取

}else{

prev = new Browser();               // 由于内存吃紧,所以对软引用的对象回收了

sr = new SoftReference(prev);       // 重新构建

}

这样就很好的解决了实际的问题。

软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所引用的对象被垃圾回收器回收后,Java虚拟机就会把这个软引用加入到与之关联的引用队列中

3.弱引用:特殊修饰,从缓存中查看对象是否已被回收,则重新创建

ReferenceQueue队列和 软 |  弱 | 虚 有关联,它们被回收后,JVM会把这些引用加入到与之关联的引用队列中。。。可能是消息提醒提示之类的。。。

只具有弱引用的对象拥有更短暂的生命周期

生命周期:在垃圾回收器线程扫描中,不管当前内存空间足够与否,都会回收它的内存。不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。

String str=new String("abc");

WeakReference abcWeakRef = new WeakReference(str);

str=null;

当垃圾回收器进行扫描回收时等价于:

str =null;

System.gc();

如果这个对象是偶尔的使用,并且希望在使用时随时就能获取到,但又不想影响此对象的垃圾收集,那么你应该用 WeakReference 来记住此对象。

下面的代码会让str再次变为一个强引用:

String  abc = abcWeakRef.get();

弱引用可以和一个引用队列(ReferenceQueue)联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。

当你想引用一个对象,但是这个对象有自己的生命周期,你不想介入这个对象的生命周期,这时候你就是用弱引用。

这个引用不会在对象的垃圾回收判断中产生任何附加的影响

package com.niuh;

import java.lang.ref.Reference;

import java.lang.ref.ReferenceQueue;

import java.lang.ref.WeakReference;

import java.util.LinkedList;

publicclass ReferenceTest {

private staticReferenceQueue rq = new ReferenceQueue();

publicstaticvoid checkQueue() {

Reference extends NiuhBig> ref = null;

while ((ref = rq.poll()) != null) {

if (ref != null) {

System.out.println("In queue: "+ ((NiuhBigWeakReference) (ref)).id);

}

}

}

publicstaticvoid main(String args[]) {

intsize= 3;

LinkedList> weakList = new LinkedList>();

for(inti = 0; i 

weakList.add(new NiuhBigWeakReference(new NiuhBig("Weak "+ i), rq));

System.out.println("Just created weak: "+ weakList.getLast());

}

System.gc();

try { // 下面休息几分钟,让上面的垃圾回收线程运行完成

Thread.currentThread().sleep(6000);

} catch (InterruptedException e) {

e.printStackTrace();

}

checkQueue();

}

}

class NiuhBig {

publicString id;

// 占用空间,让线程进行回收

byte[] b = new byte[2 * 1024];

publicNiuhBig(String id) {

this.id = id;

}

protected void finalize() {

System.out.println("Finalizing NiuhBig "+ id);

}

}

class NiuhBigWeakReference extends WeakReference {

publicString id;

publicNiuhBigWeakReference(NiuhBig big, ReferenceQueue rq) {

super(big, rq);

this.id = big.id;

}

protected void finalize() {

System.out.println("Finalizing NiuhBigWeakReference "+ id);

}

}

最后的输出结果为:

Just created weak: com.niuh.NiuhBigWeakReference@3d075dc0

Just created weak: com.niuh.NiuhBigWeakReference@214c265e

Just created weak: com.niuh.NiuhBigWeakReference@448139f0

Finalizing NiuhBig Weak 2

Finalizing NiuhBig Weak 1

Finalizing NiuhBig Weak 0

Inqueue: Weak 2

Inqueue: Weak 1

Inqueue: Weak 0

4.虚引用

生命周期:虚引用并不会决定对象的生命周期,如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。

虚引用主要用来跟踪对象被垃圾回收器回收的活动。虚引用与软引用和弱引用的一个区别在于:虚引用必须和引用队列(ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之 关联的引用队列中。 

5. 总结

当垃圾回收器回收时,某些对象会被回收,某些不会被回收。垃圾回收器会从根对象Object来标记存活的对象,然后将某些不可达的对象和一些引用的对象进行回收,通过表格来说明一下,如下:

233054D17-2.png

在实际程序设计中一般很少使用弱引用与虚引用,使用软引用(根据内存不足情况,缓存对象是否可复用)的情况较多,这是因为软引用可以加速JVM对垃圾内存的回收速度,可以维护系统的运行安全,防止内存溢出(OutOfMemory)等问题的产生。

  • 11
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java 有四种不同类型的引用: 1. 强引用(Strong Reference):通常使用的引用方式,使用强引用的对象在内存空间不足时不会被回收。 2. 软引用(Soft Reference):当内存空间不足时,对象可能会被回收,但是在回收之前它可以被缓存。 3. 弱引用(Weak Reference):当内存空间不足时,对象很快就会被回收。 4. 虚引用(Phantom Reference):它不能单独使用,必须和引用队列(ReferenceQueue)一起使用,当对象被回收时,JVM会把它加入到引用队列中。 这四种引用的不同在于,它们对垃圾回收器的回收策略不同,因此它们对于内存管理的影响不同。 ### 回答2: Java四大引用分别是强引用、软引用、弱引用和虚引用。 1. 强引用Java中最常见的引用类型。通过 `new` 关键字生成的对象默认都是强引用。强引用的对象不会被垃圾回收器回收,即使内存不足时也不会被回收。只有当强引用失去引用时,垃圾回收器才会将其回收。 2. 软引用Java中的软引用是用来描述一些还有用但不是必需的对象。在内存不足时,垃圾回收器会选择性地回收软引用对象。这种引用常常用于缓存数据,当内存足够时,可以提供数据的快速访问,当内存不足时,可以被回收释放内存。 3. 弱引用Java中的弱引用是比软引用更弱一些的引用类型。弱引用对象在垃圾回收时,只要被垃圾回收器发现,就会被回收。弱引用通常用于跟踪对象的状态,当对象被回收时,可以执行特定的操作。 4. 虚引用Java中的虚引用是最弱的一种引用。虚引用主要用于对象回收跟踪。与弱引用不同的是,虚引用在任何时候都可能被垃圾回收器回收,甚至无法通过虚引用获取对象。虚引用一般与引用队列(Reference Queue)联合使用,用于监控对象被垃圾回收的状态。 引用的不同类型在内存管理中起到不同的作用,能够提供更灵活的内存控制机制。程序员可以根据对象的生命周期和内存需求,选择合适的引用类型,从而更好地管理内存。 ### 回答3: Java四大引用是强引用(Strong Reference)、软引用(Soft Reference)、弱引用(Weak Reference)和虚引用(Phantom Reference)。 1. 强引用是默认的引用类型,也是最常用的引用类型。当对象被一个强引用引用时,对象将会一直存在,不会被垃圾回收器回收。 2. 软引用用来描述一些还有用但并非必须的对象。当内存不足时,垃圾回收器可能会回收软引用对象所占用的内存。可以通过SoftReference类来创建软引用。 3. 弱引用用来描述非必须的对象,它的生命周期比软引用更短。当垃圾回收器执行垃圾回收时,无论内存是否足够,都会回收弱引用对象。可以通过WeakReference类来创建弱引用。 4. 虚引用是最弱的引用类型,几乎没有实质性的作用。用来跟踪对象被垃圾回收器回收的活动。无法通过虚引用获得对象的实例,其主要作用是在对象被回收前收到一个系统通知。可以通过PhantomReference类来创建虚引用。 在Java中,使用合适的引用类型可以更好地管理内存,避免内存溢出。强引用应该谨慎使用,避免产生内存泄漏。而软引用和弱引用则可以用来优化缓存机制,当内存紧张时自动释放缓存。虚引用则主要用于追踪对象的回收情况。理解并正确使用这四大引用可以提高代码的性能和健壮性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值