先上结论:对于同一个类ClassA,类中有个方法MethodA,如果是单列对象,则对象是共享的,方法MethodA也是共享的,因此存在线程安全隐患;而非单列的情况则产生多个对象,方法也不是共享的,每个对象都有自己的MethodA,这样方法之前不会存在安全问题。
下面是我定时报表产生问题的例子:
如果多线程使用同一个对象A(即单列),那么多个线程则同时使用A对象中的方法,这就会导致线程安全性问题。如项目中报表发送就存在这样的问题:RepsSubscribeServiceImpl.service获得的是一个静态的对象,是单列的,因此handle方法如果不使用synchronized的话,会多个线程同时调用handle方法,同时调用就会存在安全性问题,这个已经遇到过了。
//多线程消费
final ThreadPoolExecutor tpe = new ThreadPoolExecutor(5,10,0L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<Runnable>());
public void resolve(Serializable objectMessage) throws Exception {
final RepsSubscribe bean = (RepsSubscribe)objectMessage;
log.info("开始消息处理");
//多线程消息发送
tpe.submit(new Runnable() {
public void run() {
try {
RepsSubscribeServiceImpl.service.handle(bean);
} catch (Exception e) {
log.error("报表订阅消费异常:",e);
}
}
});
}
所以,如果要多线程并且安全,则不能使用单列,必须要new一个对象:然后每个对象就会调用自己的方法,多个对象就会调用各自的方法,相当于docker的隔离机制。修改后的代码如下:
//多线程消费
final ThreadPoolExecutor tpe = new ThreadPoolExecutor(5,10,0L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<Runnable>());
public void resolve(Serializable objectMessage) throws Exception {
final RepsSubscribe bean = (RepsSubscribe)objectMessage;
log.info("开始消息处理");
//多线程消息发送
tpe.submit(new Runnable() {
public void run() {
try {
//RepsSubscribeServiceImpl.service.handle(bean);
new RepsSubscribeServiceImpl().handle(bean);
} catch (Exception e) {
log.error("报表订阅消费异常:",e);
}
}
});
}
#####################################################################################
下面用代码来验证:同一个类ClassA,不同的对象,方法是不共享的,单列的话方法是共享的这个结论。