spring单例bean线程安全问题讨论

不可变对象

final 和不可变对象的关系

final 修饰的基本类型 ,只能读不能写,天然类型安全

final 修饰的对象只是对象引用不变,对象属性可以变,没有实现不可变对象

借助 final 真正实现不可变对象

  • 类使用 final 修饰(禁止子类继承,重写父类方法,添加逻辑,破坏不可变性)
  • 所有成员变量必须是 private final
  • 只在构造方法中赋值,不提供任何 setter
  • 引用类型成员,必须做防御性拷贝
java 复制代码
public User(List<String> tags) {
    // 构造时:拷贝一个新集合,不直接用外部引用
    this.tags = List.copyOf(tags);
}

返回也要返回不可变视图 ,由构造时通过 copyOf 保证不可变,或者返回时构造不可变视图(不推荐)

List.of() 和 unmodifiableList

在 Java 中,List.of()Collections.unmodifiableList() 都会让列表拒绝写操作,但它们的语义完全不同。

List.of() 是 Java 9 引入的方式,用于从零构造一个结构不可变的列表 :该列表与任何外部集合无共享引用,严格拒绝 null ,并且拥有极简的内存布局和最佳性能。需要注意的是,它只保证列表结构不可变(不能增、删、替换元素引用),并不对元素本身做深拷贝

java 复制代码
StringBuilder sb = new StringBuilder("hello");
List<StringBuilder> list = List.of(sb);
sb.append(" world");
System.out.println(list.get(0)); // "hello world"  元素内部状态仍可被外部修改

相比之下,Collections.unmodifiableList() 创建的是对现有列表的只读视图 ,它仅拒绝通过视图的写操作,不做任何拷贝。若底层源列表仍然可变,视图中的内容会随之变化

java 复制代码
List<String> source = new ArrayList<>(List.of("a", "b"));
List<String> view = Collections.unmodifiableList(source);
source.add("c");
System.out.println(view); // [a, b, c]  视图跟着变了

选用建议:若目标是构建长期不变的常量或不可变对象的内部字段,优先使用 List.of()(从零构建)或 List.copyOf()(从已有集合拷贝)。unmodifiableList() 更适合在对外暴露内部状态时包一层只读壳,但要清楚它不提供真正的隔离。

讨论 Spring Beans 线程安全问题

重点在于是否存在共享状态,而非 Spring 容器本身。

Spring 的单例 Bean 只创建一个实例,被所有请求共享。如果该实例包含可变状态(成员变量),多线程并发修改时就会产生竞争条件。

1. 线程安全的场景:无状态 Bean

如果 Bean 没有实例变量,或只有不可变(final)变量(final 引用 + 引用的对象本身也是无状态/线程安全的),就是线程安全的。

java 复制代码
@Service
public class UserService {
    // 无成员变量,或只有不可变变量
    private final UserRepository userRepository; // final,线程安全

    public User findUser(Long id) {
        return userRepository.findById(id); // 局部变量,线程安全
    }
}

2. 线程不安全的场景:有状态 Bean

如果 Bean 包含可变成员变量,多个线程同时修改就会出问题。

java 复制代码
@Service
public class UnsafeService {
    private int counter; // 可变状态,线程不安全!

    public void increment() {
        counter++; // 非原子操作,并发时结果不确定
    }
}

如何避免

1. 使用无状态 Bean

2. 使用线程安全类(AtomicInteger 等)

3. 使用 ThreadLocal(线程本地存储)

java 复制代码
@Service
public class RequestContextService {
    private final ThreadLocal<String> currentUser = new ThreadLocal<>();

    public void setUser(String user) {
        currentUser.set(user);
    }

    public String getUser() {
        return currentUser.get();
    }

    public void clear() {
        currentUser.remove(); // 请求结束后必须清理!
    }
}

4. 方法 synchronized 修饰(不推荐)

相关推荐
zc.z14 小时前
JAVA实现:纯PCM格式音频转换成BASE64
java·音视频·pcm
mask哥15 小时前
力扣算法java实现汇总整理(上)
java·算法·leetcode
Aaswk16 小时前
Java Lambda 表达式与流处理
java·开发语言·python
是宇写的啊16 小时前
Spring AOP
java·spring
万邦科技Lafite16 小时前
京东item_get接口实战案例:实时商品价格监控全流程解析
java·开发语言·数据库·python·开放api·淘宝开放平台
Mr_pyx17 小时前
Spring AI 入门教程:Java开发者的AI应用捷径
java·人工智能·spring
Zephyr_018 小时前
Leedcode算法题
java·算法
苍煜18 小时前
Java开发IO零基础吃透:BIO、NIO、同步异步、阻塞非阻塞
java·python·nio
折哥的程序人生 · 物流技术专研19 小时前
Java面试85题图解版(一):基础核心篇
java·开发语言·后端·面试
AllData公司负责人19 小时前
通过Postgresql同步到Doris,全视角演示AllData数据中台核心功能效果,涵盖:数据入湖仓,数据同步,数据处理,数据服务,BI可视化驾驶舱
java·大数据·数据库·数据仓库·人工智能·python·postgresql