第二季:4.强引用、软引用、弱引用、虚引用分别是什么?【Java面试题】

简介: 第二季:4.强引用、软引用、弱引用、虚引用分别是什么?【Java面试题】

前言


2022 10/12 16:26

路漫漫其修远兮,吾将上下而求索


本文是根据尚硅谷学习所做笔记

仅供学习交流使用,转载注明出处

推荐

尚硅谷Java大厂面试题第2季,面试必刷,跳槽大厂神器

第二季大佬总结

4.强引用、软引用、弱引用、虚引用分别是什么?

说明

本文目录前是相关视频的名字和具体视频中思维导图的名字

题目

4.强引用、软引用、弱引用、虚引用分别是什么?

Person p = new Person()

在等号的左边,就是一个对象的引用,存储在栈中

而等号右边,就是实例化的对象,存储在堆中

其实这样的一个引用关系,就被称为强引用

整体架构



72_强引用Reference

强引用(默认支持模式)

当内存不足的时候,JVM开始垃圾回收,对于强引用的对象,就算是出现了OOM也不会对该对象进行回收,打死也不回收~!

强引用是我们最常见的普通对象引用,只要还有一个强引用指向一个对象,就能表明对象还“活着”,垃圾收集器不会碰这种对象。在Java中最常见的就是强引用,把一个对象赋给一个引用变量,这个引用变量就是一个强引用。当一个对象被强引用变量引用时,它处于可达状态,它是不可能被垃圾回收机制回收的,即使该对象以后永远都不会被用到,JVM也不会回收,因此强引用是造成Java内存泄漏的主要原因之一。

对于一个普通的对象,如果没有其它的引用关系,只要超过了引用的作用于或者显示地将相应(强)引用赋值为null,一般可以认为就是可以被垃圾收集的了(当然具体回收时机还是要看垃圾回收策略)

case

package references4;
/**
 * @author CSDN@日星月云
 * @date 2022/10/12 16:37
 */
public class StrongReferenceDemo {
    public static void main(String[] args) {
        // 这样定义的默认就是强应用
        Object obj1 = new Object();
        // 使用第二个引用,指向刚刚创建的Object对象
        Object obj2 = obj1;
        // 置空
        obj1 = null;
        // 垃圾回收
        System.gc();
        System.out.println(obj1);
        System.out.println(obj2);
    }
}
null
java.lang.Object@1b6d3586

73_软引用SoftReference

软引用

软引用是一种相对弱化了一些的引用,需要用Java.lang.ref.SoftReference类来实现,可以让对象豁免一些垃圾收集,

对于只有软引用的对象来讲:
当系统内存充足时,它不会被回收

当系统内存不足时,它会被回收

软引用通常在对内存敏感的程序中,比如高速缓存就用到了软引用,内存够用的时候就保留,不够用就回收


package references4;
import java.lang.ref.SoftReference;
/**
 * @author CSDN@日星月云
 * @date 2022/10/12 21:24
 */
public class SoftReferenceDemo {
    /**
     * 内存够用的时候就保留,不够用就回收
     */
    public static void softRefMemoryEnough() {
        // 创建一个强应用
        Object o1 = new Object();
        // 创建一个软引用
        SoftReference<Object> softReference = new SoftReference<>(o1);
        System.out.println(o1);
        System.out.println(softReference.get());
        o1 = null;
        // 手动GC
        System.gc();
        System.out.println(o1);
        System.out.println(softReference.get());
    }
    /**
     * JVM配置,故意产生大对象并配置小的内存,让它的内存不够用了导致OOM,看软引用的回收情况
     * -Xms5m -Xmx5m -XX:+PrintGCDetails
     */
    public static void softRefMemoryNoEnough() {
        System.out.println("========================");
        // 创建一个强应用
        Object o1 = new Object();
        // 创建一个软引用
        SoftReference<Object> softReference = new SoftReference<>(o1);
        System.out.println(o1);
        System.out.println(softReference.get());
        o1 = null;
        // 模拟OOM自动GC
        try {
            // 创建30M的大对象
            byte[] bytes = new byte[30 * 1024 * 1024];
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            System.out.println(o1);
            System.out.println(softReference.get());
        }
    }
    public static void main(String[] args) {
//        softRefMemoryEnough();
        /*
        java.lang.Object@1b6d3586
        java.lang.Object@1b6d3586
        null
        java.lang.Object@1b6d3586
         */
        softRefMemoryNoEnough();
        /*
        java.lang.Object@1b6d3586
        java.lang.Object@1b6d3586
        GC
        null
        null
        OutOfMemoryError: Java heap space
         */
    }
}

74_弱引用WeakReference

弱引用

不管内存是否够,只要有GC操作就会进行回收

弱引用需要用 java.lang.ref.WeakReference 类来实现,它比软引用生存期更短

对于只有弱引用的对象来说,只要垃圾回收机制一运行,不管JVM的内存空间是否足够,都会回收该对象占用的空间。

case

package references4;
import java.lang.ref.WeakReference;
/**
 * @author CSDN@日星月云
 * @date 2022/10/12 21:32
 */
public class WeakReferenceDemo {
    public static void main(String[] args) {
        Object o1=new Object();
        WeakReference<Object> weakReference=new WeakReference<>(o1);
        System.out.println(o1);
        System.out.println(weakReference.get());
        o1=null;
        System.gc();
        System.out.println("========");
        System.out.println(o1);
        System.out.println(weakReference.get());
        //null
        //null
    }
}

75_软引用和弱引用的适用场景

软引用和弱引用的适用场景

假如有一个应用需要读取大量的本地图片


如果每次读取图片都从硬盘读取则会严重影响性能

如果一次性全部加载到内存中,又可能造成内存溢出

此时使用软引用可以解决这个问题


设计思路:使用HashMap来保存图片的路径和相应图片对象关联的软引用之间的映射关系,在内存不足时,JVM会自动回收这些缓存图片对象所占的空间,从而有效地避免了OOM的问题

Map<String, SoftReference<String>> imageCache = new HashMap<String, SoftReference<Bitmap>>();

76_WeakHashMap案例演示和解析

你知道弱引用的话,能谈谈WeakHashMap吗?



package references4;
import java.util.HashMap;
import java.util.WeakHashMap;
/**
 * @author CSDN@日星月云
 * @date 2022/10/12 21:41
 */
public class WeakHashMapDemo {
    public static void main(String[] args) {
        myHashMap();
        System.out.println("==========");
        myWeakHashMap();
    }
    private static void myHashMap() {
        HashMap<Integer,String> map=new HashMap<>();
        Integer key=new Integer(1);
        String value="HashMap";
        map.put(key,value);
        System.out.println(map);//{1=HashMap}
        key=null;
        System.out.println(map);//{1=HashMap}
        System.gc();
        System.out.println(map+"\t"+map.size());//{1=HashMap}   1
    }
    private static void myWeakHashMap() {
        WeakHashMap<Integer,String> map=new WeakHashMap<>();
        Integer key=new Integer(2);
        String value="WeakHashMap";
        map.put(key,value);
        System.out.println(map);//{2=WeakHashMap}
        key=null;
        System.out.println(map);//{2=WeakHashMap}
        System.gc();
        System.out.println(map+"\t"+map.size());//{}   0
    }
}

77_虚引用简介

虚引用

虚引用需要java.lang.ref.PhantomReference类来实现。
顾名思义,就是形同虚设,与其他几种引用都不同,虚引用并不会决定对象的生命周期。

如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收它不能单独使用也不能通过它访问对象,虚引用必须和引用队列(ReferenceQueue)联合使用。


虚引用的主要作用是跟踪对象被垃圾回收的状态。仅仅是提供了一种确保对象被finalize以后,做某些事情的机制。

PhantomReference的get方法总是返回nul,因此无法访问对应的引用对象。其意义在于说明一个对象已经进入finalization阶段,可以被gc回收,用来实现比finalization机制更灵活的回收操作。


换句话说,设置虚引用关联的唯一目的,就是在这个对象被收集器回收的时候收到一个系统通知或者后续添加进一步的处理。Java技术允许使用finalize()方法在垃圾收集器将对象从内存中清除出去之前做必要的清理工作。



引用队列



78_ReferenceQueue引用队列介

case

package references4;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.WeakReference;
/**
 * @author CSDN@日星月云
 * @date 2022/10/12 21:58
 */
public class ReferenceQueueDemo {
    public static void main(String[] args) throws InterruptedException {
        Object o1=new Object();
        ReferenceQueue<Object>  referenceQueue=new ReferenceQueue();
        WeakReference<Object> weakReference=new WeakReference<>(o1,referenceQueue);
        System.out.println(o1);//java.lang.Object@1b6d3586
        System.out.println(weakReference.get());//java.lang.Object@1b6d3586
        System.out.println(referenceQueue.poll());//null
        System.out.println("=========");
        o1=null;
        System.gc();
        Thread.sleep(500);
        System.out.println(o1);//null
        System.out.println(weakReference.get());//null
        System.out.println(referenceQueue.poll());//java.lang.ref.WeakReference@4554617c
    }
}

从这里我们能看到,在进行垃圾回收后,我们弱引用对象,也被设置成null,但是在队列中还能够导出该引用的实例,这就说明在回收之前,该弱引用的实例被放置引用队列中了,我们可以通过引用队列进行一些后置操作

79_虚引用PhantomReference

case

package references4;
import java.lang.ref.PhantomReference;
import java.lang.ref.ReferenceQueue;
/**
 * @author CSDN@日星月云
 * @date 2022/10/12 22:03
 * Java提供了4种川用类型,在垃圾何收的时候,都有自己各自的特点.
 * ReferenceQueue是用来配合引用工作的,没有ReferenceQueue一样可以运行。
 *
 * 创建引用的时候可以指定关联的队列,当GC释放对象内存的时候,会将引用加入到引用队列,
 * 如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取必要的行动
 * 这相当于是一种通知机制。
 *
 * 当关联的引用队列中有数据的时候,意味着引用指向的堆内存中的对象被回收。通过这种方式,JWN允许我们在对象被销毁后,
 * 做一些我们自己想做的事情。
 *
 */
public class PhantomReferenceDemo {
    public static void main(String[] args) throws InterruptedException {
        Object o1=new Object();
        ReferenceQueue<Object> referenceQueue=new ReferenceQueue();
        PhantomReference<Object> phantomReference=new PhantomReference<>(01,referenceQueue);
        System.out.println(o1);//java.lang.Object@1b6d3586
        System.out.println(phantomReference.get());//null
        System.out.println(referenceQueue.poll());//null
        System.out.println("=================");
        o1=null;
        System.gc();
        Thread.sleep(500);
        System.out.println(o1);//null
        System.out.println(phantomReference.get());//null
        System.out.println(referenceQueue.poll());//hashcode null
    }
}

80_GCRoots和四大引用小总结

GCRoots和四大引用小总结

红色部分在垃圾回收之外,也就是强引用的

蓝色部分:属于软引用,在内存不够的时候,才回收

虚引用和弱引用:每次垃圾回收的时候,都会被干掉,但是它在干掉之前还会存在引用队列中,我们可以通过引用队列进行一些通知机制


最后


2022 10/12 22:13


p72~p80


Markdown 7447 字数 462 行数

HTML 6961 字数 274 段落

相关文章
|
4月前
|
Java 测试技术 微服务
最新技术栈下 Java 面试高频技术点实操指南详解
本指南结合最新Java技术趋势,涵盖微服务(Spring Cloud Alibaba)、响应式编程(Spring WebFlux)、容器化部署(Docker+Kubernetes)、函数式编程、性能优化及测试等核心领域。通过具体实现步骤与示例代码,深入讲解服务注册发现、配置中心、熔断限流、响应式数据库访问、JVM调优等内容。适合备战Java面试,提升实操能力,助力技术进阶。资源链接:[https://pan.quark.cn/s/14fcf913bae6](https://pan.quark.cn/s/14fcf913bae6)
170 25
|
4月前
|
缓存 Java 关系型数据库
2025 年最新华为 Java 面试题及答案,全方位打造面试宝典
Java面试高频考点与实践指南(150字摘要) 本文系统梳理了Java面试核心考点,包括Java基础(数据类型、面向对象特性、常用类使用)、并发编程(线程机制、锁原理、并发容器)、JVM(内存模型、GC算法、类加载机制)、Spring框架(IoC/AOP、Bean生命周期、事务管理)、数据库(MySQL引擎、事务隔离、索引优化)及分布式(CAP理论、ID生成、Redis缓存)。同时提供华为级实战代码,涵盖Spring Cloud Alibaba微服务、Sentinel限流、Seata分布式事务,以及完整的D
202 2
|
4月前
|
存储 安全 Java
常见 JAVA 集合面试题整理 自用版持续更新
这是一份详尽的Java集合面试题总结,涵盖ArrayList与LinkedList、HashMap与HashTable、HashSet与TreeSet的区别,以及ConcurrentHashMap的实现原理。内容从底层数据结构、性能特点到应用场景逐一剖析,并提供代码示例便于理解。此外,还介绍了如何遍历HashMap和HashTable。无论是初学者还是进阶开发者,都能从中受益。代码资源可从[链接](https://pan.quark.cn/s/14fcf913bae6)获取。
222 3
|
3月前
|
缓存 Java API
Java 面试实操指南与最新技术结合的实战攻略
本指南涵盖Java 17+新特性、Spring Boot 3微服务、响应式编程、容器化部署与数据缓存实操,结合代码案例解析高频面试技术点,助你掌握最新Java技术栈,提升实战能力,轻松应对Java中高级岗位面试。
336 0
|
4月前
|
存储 安全 Java
2025 最新史上最全 Java 面试题独家整理带详细答案及解析
本文从Java基础、面向对象、多线程与并发等方面详细解析常见面试题及答案,并结合实际应用帮助理解。内容涵盖基本数据类型、自动装箱拆箱、String类区别,面向对象三大特性(封装、继承、多态),线程创建与安全问题解决方法,以及集合框架如ArrayList与LinkedList的对比和HashMap工作原理。适合准备面试或深入学习Java的开发者参考。附代码获取链接:[点此下载](https://pan.quark.cn/s/14fcf913bae6)。
1418 48
|
4月前
|
算法 架构师 Java
Java 开发岗及 java 架构师百度校招历年经典面试题汇总
以下是百度校招Java岗位面试题精选摘要(150字): Java开发岗重点关注集合类、并发和系统设计。HashMap线程安全可通过Collections.synchronizedMap()或ConcurrentHashMap实现,后者采用分段锁提升并发性能。负载均衡算法包括轮询、加权轮询和最少连接数,一致性哈希可均匀分布请求。Redis持久化有RDB(快照恢复快)和AOF(日志更安全)两种方式。架构师岗涉及JMM内存模型、happens-before原则和无锁数据结构(基于CAS)。
116 5
|
4月前
|
Java API 微服务
2025 年 Java 校招面试全攻略:从面试心得看 Java 岗位求职技巧
《2025年Java校招最新技术要点与实操指南》 本文梳理了2025年Java校招的核心技术栈,并提供了可直接运行的代码实例。重点技术包括: Java 17+新特性(Record类、Sealed类等) Spring Boot 3+WebFlux响应式编程 微服务架构与Spring Cloud组件 Docker容器化部署 Redis缓存集成 OpenAI API调用 通过实际代码演示了如何应用这些技术,如Java 17的Record类简化POJO、WebFlux构建响应式API、Docker容器化部署。
153 5
|
4月前
|
缓存 NoSQL Java
Java Redis 面试题集锦 常见高频面试题目及解析
本文总结了Redis在Java中的核心面试题,包括数据类型操作、单线程高性能原理、键过期策略及分布式锁实现等关键内容。通过Jedis代码示例展示了String、List等数据类型的操作方法,讲解了惰性删除和定期删除相结合的过期策略,并提供了Spring Boot配置Redis过期时间的方案。文章还探讨了缓存穿透、雪崩等问题解决方案,以及基于Redis的分布式锁实现,帮助开发者全面掌握Redis在Java应用中的实践要点。
201 6
|
4月前
|
安全 Java API
2025 年 Java 校招面试常见问题及详细答案汇总
本资料涵盖Java校招常见面试题,包括Java基础、并发编程、JVM、Spring框架、分布式与微服务等核心知识点,并提供详细解析与实操代码,助力2025校招备战。
200 1
|
4月前
|
算法 Java 微服务
2025 年 Java 面试宝典社招春招秋招实操全方位攻略
2025年Java面试宝典涵盖核心技术及最新趋势,分为四大板块:1. Java基础:深入数据类型、多态等特性,结合学生信息管理等实例;2. JVM核心:解析内存模型与GC算法,附多线程转账等场景应用;3. 高并发方案:详解synchronized与线程池配置,提供Web服务器优化案例;4. Spring生态:剖析IoC/AOP原理,演示微服务架构实现。特别新增Java 17+特性实操,包括Record类、密封接口等语法糖,整合Spring Boot 3、响应式编程及云原生技术,通过订单状态机、API网关配置。
266 1

热门文章

最新文章