news 2026/4/18 8:54:20

每日Java面试场景题知识点之-单例模式

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
每日Java面试场景题知识点之-单例模式

每日Java面试场景题知识点之-单例模式

一、单例模式概述

单例模式(Singleton Pattern)是Java中最简单也是最常用的设计模式之一。它保证一个类只有一个实例,并提供一个全局访问点来访问这个实例。在Java企业级项目中,单例模式广泛应用于配置管理、数据库连接池、缓存管理等场景。

二、单例模式的多种实现方式

1. 饿汉式单例模式

public class EagerSingleton { // 在类加载时就创建实例,线程安全但可能造成资源浪费 private static final EagerSingleton instance = new EagerSingleton(); private EagerSingleton() {} public static EagerSingleton getInstance() { return instance; } }

特点:

  • 线程安全
  • 类加载时就创建实例
  • 可能造成资源浪费

2. 懒汉式单例模式(线程不安全)

public class LazySingleton { private static LazySingleton instance; private LazySingleton() {} public static LazySingleton getInstance() { if (instance == null) { instance = new LazySingleton(); } return instance; } }

**问题:**在多线程环境下,可能创建多个实例。

3. 懒汉式单例模式(线程安全)

public class ThreadSafeLazySingleton { private static ThreadSafeLazySingleton instance; private ThreadSafeLazySingleton() {} public static synchronized ThreadSafeLazySingleton getInstance() { if (instance == null) { instance = new ThreadSafeLazySingleton(); } return instance; } }

特点:

  • 线程安全
  • 性能较差(每次获取实例都要同步)

4. 双重检查锁定(DCL)

public class DCLSingleton { private static volatile DCLSingleton instance; private DCLSingleton() {} public static DCLSingleton getInstance() { if (instance == null) { synchronized (DCLSingleton.class) { if (instance == null) { instance = new DCLSingleton(); } } } return instance; } }

特点:

  • 线程安全
  • 性能较好
  • 需要使用volatile关键字防止指令重排序

5. 静态内部类方式

public class InnerClassSingleton { private InnerClassSingleton() {} private static class SingletonHolder { private static final InnerClassSingleton INSTANCE = new InnerClassSingleton(); } public static InnerClassSingleton getInstance() { return SingletonHolder.INSTANCE; } }

特点:

  • 线程安全
  • 延迟加载
  • 性能最好

6. 枚举方式

public enum EnumSingleton { INSTANCE; public void doSomething() { // 业务逻辑 } }

特点:

  • 线程安全
  • 防止反射攻击
  • 防止序列化破坏

三、企业级项目中的实际应用场景

1. 配置管理器

在大型企业应用中,配置信息通常需要全局共享且只加载一次。

public class ConfigManager { private static volatile ConfigManager instance; private Properties properties; private ConfigManager() { properties = new Properties(); loadConfig(); } public static ConfigManager getInstance() { if (instance == null) { synchronized (ConfigManager.class) { if (instance == null) { instance = new ConfigManager(); } } } return instance; } private void loadConfig() { try { properties.load(ConfigManager.class.getClassLoader().getResourceAsStream("config.properties")); } catch (IOException e) { throw new RuntimeException("加载配置文件失败", e); } } public String getProperty(String key) { return properties.getProperty(key); } }

2. 数据库连接池

数据库连接池是单例模式的典型应用,确保全局只有一个连接池实例。

public class ConnectionPool { private static volatile ConnectionPool instance; private DataSource dataSource; private ConnectionPool() { initDataSource(); } public static ConnectionPool getInstance() { if (instance == null) { synchronized (ConnectionPool.class) { if (instance == null) { instance = new ConnectionPool(); } } } return instance; } private void initDataSource() { HikariConfig config = new HikariConfig(); config.setJdbcUrl("jdbc:mysql://localhost:3306/mydb"); config.setUsername("root"); config.setPassword("password"); config.setMaximumPoolSize(20); dataSource = new HikariDataSource(config); } public Connection getConnection() throws SQLException { return dataSource.getConnection(); } }

3. 缓存管理器

在电商系统中,缓存管理器通常需要全局唯一实例。

public class CacheManager { private static volatile CacheManager instance; private Cache<String, Object> cache; private CacheManager() { cache = Caffeine.newBuilder() .maximumSize(1000) .expireAfterWrite(10, TimeUnit.MINUTES) .build(); } public static CacheManager getInstance() { if (instance == null) { synchronized (CacheManager.class) { if (instance == null) { instance = new CacheManager(); } } } return instance; } public void put(String key, Object value) { cache.put(key, value); } public Object get(String key) { return cache.getIfPresent(key); } public void remove(String key) { cache.invalidate(key); } }

四、常见问题及解决方案

1. 线程安全问题

**问题:**在多线程环境下,单例模式可能创建多个实例。

解决方案:

  • 使用synchronized关键字
  • 使用双重检查锁定(DCL)
  • 使用静态内部类
  • 使用枚举

2. 反射攻击问题

**问题:**通过反射可以破坏单例模式。

解决方案:

private Singleton() { if (instance != null) { throw new RuntimeException("单例模式不允许反射创建实例"); } }

3. 序列化破坏问题

**问题:**通过序列化和反序列化可以破坏单例模式。

解决方案:

protected Object readResolve() { return getInstance(); }

4. 性能问题

**问题:**同步方法会影响性能。

解决方案:

  • 使用双重检查锁定
  • 使用静态内部类
  • 使用枚举

五、最佳实践

  1. 选择合适的实现方式:根据具体场景选择最适合的单例实现方式
  2. 考虑线程安全:在多线程环境中必须保证线程安全
  3. 防止反射和序列化破坏:对于重要的单例类,需要防止反射和序列化破坏
  4. 考虑资源管理:确保单例类能够正确管理资源
  5. 避免过度使用:单例模式虽然方便,但过度使用会导致代码耦合度高

六、总结

单例模式是Java企业级开发中的重要设计模式,它保证一个类只有一个实例,并提供全局访问点。在实际项目中,我们需要根据具体场景选择合适的实现方式,并注意线程安全、反射攻击、序列化破坏等问题。

掌握单例模式对于Java开发者来说是非常重要的,它不仅能够帮助我们写出更好的代码,还能够在面试中展示我们的设计能力。

感谢读者观看!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/17 22:00:56

每日Java面试场景题知识点之-工厂方法模式

每日Java面试场景题知识点之-工厂方法模式 场景问题描述 在开发一个电商平台支付系统时&#xff0c;技术团队面临以下挑战&#xff1a; 系统需要集成多种支付渠道&#xff0c;包括支付宝、微信支付、银联支付等。最初采用硬编码方式实现支付逻辑&#xff0c;每次新增支付渠道都…

作者头像 李华
网站建设 2026/4/18 1:59:45

Wan2.2-T2V-A14B支持分镜脚本输入生成吗?未来规划

Wan2.2-T2V-A14B 支持分镜脚本输入吗&#xff1f;我们离“AI拍电影”还有多远 &#x1f3ac; 你有没有试过这样一种场景&#xff1a; 写好一个故事板&#xff0c;点一下按钮&#xff0c;AI 就自动帮你生成一部像模像样的短片——镜头推拉、角色走位、情绪节奏全都对味儿。听起来…

作者头像 李华
网站建设 2026/4/17 20:56:39

终极指南:5分钟快速上手Parse Dashboard完整部署与配置

终极指南&#xff1a;5分钟快速上手Parse Dashboard完整部署与配置 【免费下载链接】parse-dashboard A dashboard for managing Parse Server 项目地址: https://gitcode.com/gh_mirrors/pa/parse-dashboard Parse Dashboard作为Parse Server的官方数据管理界面&#x…

作者头像 李华
网站建设 2026/4/17 20:30:03

腾讯云SA9和S9适用场景,以及价格分析

腾讯云SA9和S9适用场景 SA9实例适用场景 SA9实例基于AMD EPYC处理器&#xff0c;适合计算密集型任务。典型场景包括高性能计算&#xff08;HPC&#xff09;、大数据分析、视频编码等需要高并行计算能力的场景。其多核心设计对多线程应用优化明显&#xff0c;如科学计算、3D渲染…

作者头像 李华
网站建设 2026/4/18 8:17:31

芯片设计岗位:核心技能与实战案例详解

芯片设计是高度专业化、流程化的领域&#xff0c;不同细分岗位&#xff08;前端 / 验证 / 后端 / 模拟 IC/DFT&#xff09;的技能要求差异显著&#xff0c;但 “基础理论 工具实操 流程把控” 是所有岗位的通用门槛。本文从 “岗位分类→通用核心技能→专项技能 案例→工程实…

作者头像 李华