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 修饰(不推荐)

相关推荐
周末也要写八哥几秒前
C++实际开发之泛型编程(模版编程)
java·开发语言·c++
好家伙VCC1 分钟前
**发散创新:用 Rust实现游戏日引擎核心模块——从事件驱动到多线程调度的实战
java·开发语言·python·游戏·rust
一个有温度的技术博主1 分钟前
Spring Cloud 入门与实战:从架构拆分到核心组件详解
spring·spring cloud·架构
014-code6 分钟前
Chronicle Queue:把 Disruptor 的数据落盘
java·服务器
小江的记录本13 分钟前
【系统设计】《2026高频经典系统设计题》(秒杀系统、短链接系统、订单系统、支付系统、IM系统、RAG系统设计)(完整版)
java·后端·python·安全·设计模式·架构·系统架构
希望永不加班21 分钟前
SpringBoot 中 AOP 实现权限校验(角色/权限)
java·spring boot·后端·spring
桌面运维家33 分钟前
IDV云桌面vDisk机房部署方案模板特性解析
java·开发语言·devops
哈密瓜刨冰1 小时前
深入浅出 SpringMVC:核心注解全解析与实战用法
java
geNE GENT2 小时前
Spring Boot管理用户数据
java·spring boot·后端
怒放吧德德2 小时前
Spring Boot实战:Event事件机制解析与实战
java·spring boot·后端