走查代码时发现有dubbo接口,将入参作为了类的成员变量。Spring内大多数注释默认是单例的,也就是说这样形成了一个有状态的单例。
在单线程环境,或者入参固定的场景下,这样是没有问题的,但是在并发情况下,成员变量的获取可能会有问题。
问题代码模拟
为了方便使用
@Service
模拟问题接口:
@Service
public class SingletonStateTestService {
private static final Logger logger = LoggerFactory.getLogger(SingletonStateTestService.class);
private int serviceState; // 成员变量,可能有并发问题
public void stateTest(int state) throws InterruptedException {
logger.info("request state is {}", state);
serviceState = state;
Thread.sleep(1000);
logger.info("serviceState is {}", serviceState);
}
}
模拟多线程调用:
for (int i = 0; i < 5; i++) {
int finalI = i;
// 使用5各线程,分别传入不同的入参
new Thread(() ->{
try {
service.stateTest(finalI);
} catch (InterruptedException e) {
e.printStackTrace();
}
}).start();
}
实际执行结果如下:
入参的5个state各不同,但是打印的成员变量serviceState却有重复的。
预期的结果应该为:每个接口将传入的state赋值给成员变量serviceState,每个接口打印不同的serviceState。
造成原因
Spring大多数的bean默认都是单例Bean,单例的类全局只有一个实例。也就是说测试接口只有一个实例,每次调用时,入参的state都会覆盖之前的设置。在并发场景下,可能存在
后面的线程覆盖
了前面未结束线程设置的成员变量,导致
前面的线程获取的成员变量不正确
。
解决方法
单例中使用成员变量可能造成并发问题,若仍需要保持使用单例模式的话,需要
去掉成员变量
。
1. 成员变量改为方法的内部参数
将成员变量去掉,需要使用的部分作为方法参数带着。
2. 使用TreadLocal
若需要使用的方法太多,不方便所有的方法都加参数。可以使用TreadLocal。
@Service
public class SingletonStateTestService {
private static final Logger logger = LoggerFactory.getLogger(SingletonStateTestService.class);
private static final ThreadLocal<Integer> STATE = new NamedThreadLocal<>("STATE");
public void stateTest(int state) throws InterruptedException {
logger.info("request state is {}", state);
try {
STATE.set(state);
Thread.sleep(1000);
logger.info("serviceState is {}", STATE.get());
}
finally {
STATE.remove();
}
}
}
每个线程都用自己维护自己的变量,不会造成并发问题。
返回结果: