Spring 的 Bean 是否线程安全核心答案是不一定这完全取决于 Bean 的作用域Scope和自身的状态设计而非 Spring 框架本身提供了线程安全保障。一、核心原理作用域决定基础行为Spring 中最常用的 Bean 作用域是singleton单例默认和prototype原型二者的线程安全特性差异极大1. 单例 Beansingleton默认非线程安全行为Spring 容器启动时创建唯一实例整个应用生命周期内复用这个实例所有线程共享同一个 Bean 对象。线程安全问题根源如果 Bean 包含可变的成员变量状态多个线程同时读写这些变量时会出现线程安全问题如数据脏读、竞态条件。示例非线程安全的单例 Beanimport org.springframework.stereotype.Component; // 单例 Bean默认scopesingleton Component public class NonThreadSafeBean { // 可变状态多个线程共享此变量 private int count 0; // 无同步措施的写操作线程不安全 public void increment() { count; // 多线程下会出现计数错误 } public int getCount() { return count; } }测试验证多线程调用import org.springframework.context.annotation.AnnotationConfigApplicationContext; public class Test { public static void main(String[] args) throws InterruptedException { AnnotationConfigApplicationContext context new AnnotationConfigApplicationContext(); context.scan(com.example); context.refresh(); NonThreadSafeBean bean context.getBean(NonThreadSafeBean.class); // 1000个线程同时调用increment for (int i 0; i 1000; i) { new Thread(bean::increment).start(); } Thread.sleep(1000); System.out.println(最终计数 bean.getCount()); // 结果大概率小于1000线程不安全 } }2. 原型 Beanprototype线程安全前提是不共享行为每次从容器获取 Bean 时Spring 都会创建一个新的实例每个线程拿到的是独立的对象。线程安全特性只要线程不共享这个原型 Bean 实例比如每个线程自己获取、自己使用就不存在线程安全问题但如果手动将原型 Bean 共享给多个线程依然会有问题。示例线程安全的原型 Beanimport org.springframework.context.annotation.Scope; import org.springframework.stereotype.Component; import static org.springframework.beans.factory.config.ConfigurableBeanFactory.SCOPE_PROTOTYPE; Component Scope(SCOPE_PROTOTYPE) // 原型作用域 public class ThreadSafePrototypeBean { private int count 0; public void increment() { count; } public int getCount() { return count; } }测试验证每个线程获取独立实例public class Test { public static void main(String[] args) throws InterruptedException { AnnotationConfigApplicationContext context new AnnotationConfigApplicationContext(); context.scan(com.example); context.refresh(); // 1000个线程每个线程获取独立的原型Bean for (int i 0; i 1000; i) { new Thread(() - { ThreadSafePrototypeBean bean context.getBean(ThreadSafePrototypeBean.class); bean.increment(); System.out.println(当前线程计数 bean.getCount()); // 结果恒为1线程安全 }).start(); } } }3. 其他作用域补充request每个 HTTP 请求创建一个 Bean仅在当前请求线程内有效线程安全。session每个用户会话创建一个 Bean仅在会话所属线程内有效线程安全但会话内多线程访问仍需注意。application同 singleton全局单例非线程安全。二、如何保证单例 Bean 的线程安全如果必须使用单例 Bean大多数场景推荐单例性能更高可通过以下方式解决线程安全问题无状态设计推荐让 Bean 不包含任何可变成员变量仅包含方法无状态比如纯工具类、Controller仅转发请求无自身状态。示例Component public class StatelessUtilBean { // 无成员变量仅提供方法天然线程安全 public int add(int a, int b) { return a b; } }同步机制对可变状态的读写操作加锁synchronized、ReentrantLock但会牺牲性能。示例Component public class ThreadSafeSingletonBean { private int count 0; // 加synchronized保证原子性 public synchronized void increment() { count; } public synchronized int getCount() { return count; } }使用线程安全的容器用java.util.concurrent包下的线程安全类如AtomicInteger、ConcurrentHashMap替代普通变量。示例Component public class ThreadSafeSingletonBean { // AtomicInteger保证原子操作线程安全 private AtomicInteger count new AtomicInteger(0); public void increment() { count.incrementAndGet(); } public int getCount() { return count.get(); } }ThreadLocal 隔离线程状态为每个线程分配独立的变量副本线程间互不干扰适合需要线程内共享状态的场景。示例Component public class ThreadLocalBean { // ThreadLocal为每个线程保存独立的count副本 private ThreadLocalInteger count ThreadLocal.withInitial(() - 0); public void increment() { count.set(count.get() 1); } public int getCount() { return count.get(); } }总结Spring Bean 的线程安全不取决于框架本身核心是 Bean 的作用域和状态设计单例 Bean 因共享实例可变状态会导致线程不安全原型 /request/session 作用域 Bean 因实例隔离天然线程安全除非手动共享。保证单例 Bean 线程安全的最优方案是无状态设计其次可通过同步锁、线程安全容器、ThreadLocal 等方式解决。实际开发中优先使用单例 Bean性能更高仅在必要时使用原型 Bean且避免在单例 Bean 中存放可变的成员变量。