场景描述:
A类的for循环中使用多线程调用B的方法,假如B类中含有成员变量,那么想要保证线程安全,那么每次A调用B的时候必须保证调用到的是不同的B对象,也就用到了prototype。
实现步骤:
1.将B的scope配置为prototype。(因为自动注入的原因导致B还是单例的)
2.A实现BeanFactoryAware 接口,重写setBeanFactory方法
3.通过beanFactory.getBean("这里填写bean的id")
这样获得的B对象就是不同的对象了。
示例:
第1步.将B的scope配置为prototype。
<bean id="b" class="com.abc.xpy.B" scope="prototype"/>
<bean id="a" class="com.abc.xpy.A"/>
第2、3步:A实现BeanFactoryAware ,重写setBeanFactory方法,通过beanFactory.getBean("这里填写bean的id")
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
public class A implements BeanFactoryAware {
//定义BeanFactory
private BeanFactory beanFactory;
/**
* ThreadPool 自定义一个线程池
*/
private static final ThreadPoolExecutor EXECUTOR = new ThreadPoolExecutor(2, 4, 60L, TimeUnit.SECONDS, new LinkedBlockingQueue<>(60000));
public void test() {
//自己的数组或集合,这里不再进行填充
String[] ids = new String[10]
for (String id : ids) {
EXECUTOR.execute(new Runnable() {
@Override
public void run() {
try {
//这里每次线程过来,b就是不同的对象了
B b = (B) beanFactory.getBean("b");
b.test(id);
} catch (Exception e) {
//todo
}
}
});
}
}
@Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
this.beanFactory = beanFactory;
}
}
不知出于什么原因,试过一些主流的解决方法都不行,可能是因为框架原因(某大厂外包),这个亲测有效!!