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

相关推荐
我是大猴子2 小时前
事务失效的几种情况以及是为什么(详解)
java·开发语言
wertyuytrewm3 小时前
Java面试——Java基础
java·jvm·面试
czlczl200209253 小时前
RAG实现思路流程
java·jvm
带娃的IT创业者3 小时前
WeClaw_40_系统监控与日志体系:多层次日志架构与Trace追踪
java·开发语言·python·架构·系统监控·日志系统·链路追踪
Y001112363 小时前
JDBC原理
java·开发语言·数据库·jdbc
程序员侠客行4 小时前
Tomcat 从陌生到熟悉
java·tomcat·web
wertyuytrewm4 小时前
Java 异常|Java Exceptions
java·开发语言
ProgramHelpOa4 小时前
Amazon SDE Intern OA 2026 最新复盘|70分钟两题 Medium-Hard
java·前端·javascript
雪碧聊技术4 小时前
深入理解 Java GC:从“房间清洁工”到解决系统卡顿实战
java·开发语言