为了账号安全,请及时绑定邮箱和手机立即绑定

一段java并发同步示例代码的疑惑

一段java并发同步示例代码的疑惑

慕森卡 2019-04-19 16:12:30
importjava.util.ArrayList;importjava.util.concurrent.Callable;importjava.util.concurrent.ExecutionException;importjava.util.concurrent.ExecutorService;importjava.util.concurrent.Executors;importjava.util.concurrent.Future;publicclassCachedThreadPool{privatestaticintid=0;publicstaticvoidmain(String[]args){newCachedThreadPool().fun();}privatevoidfun(){ExecutorServiceexe=Executors.newCachedThreadPool();ArrayListlist=newArrayList();for(inti=0;i
查看完整描述

2 回答

?
炎炎设计

TA贡献1808条经验 获得超4个赞

因为放到TaskCall里之后,synchronized表示在一个TaskCall实例上同步执行。有3个实例,它们之间是不同步的。
而放在外面是在一个CachedThreadPool中同步。
                            
查看完整回答
反对 回复 2019-04-19
?
智慧大石

TA贡献1946条经验 获得超3个赞

按你的测试代码,楼上回答能解释。
然而:
测试代码有错误,以下是正确的测试代码:
importjava.util.ArrayList;
importjava.util.Collections;
importjava.util.List;
importjava.util.concurrent.*;
publicclassCachedThreadPool{
privatestaticintid=0;
privatestaticListresults=Collections.synchronizedList(newArrayList<>());
publicstaticvoidmain(String[]args)throwsException{
newCachedThreadPool().fun();
}
privatevoidfun()throwsException{
ExecutorServiceexe=Executors.newCachedThreadPool();
List>list=newArrayList<>();
for(inti=0;i<100;i++){
list.add(exe.submit(newTaskCall()));
}
exe.shutdown();
for(Futurefs:list){
fs.get();
}
//断言
for(inti=1;iif(results.get(i)!=results.get(i-1)+1){
thrownewIllegalStateException();
}
}
System.out.println("\n"+results);
}
privatesynchronizedStringgetId(){
++id;
System.out.print(id+",");
results.add(id);
returnid+"";
}
classTaskCallimplementsCallable{
@Override
publicStringcall()throwsException{
returngetId();
}
}
}
然后是分析:
先翻译几处代码,以便理解:
1.
privatesynchronizedStringgetId(){
return++id+"";
}
等价于
privateStringgetId(){
synchronized(this){
return++id+"";
}
}
2.
//TaskCall
returngetId();
等价于
//TaskCall
returnCachedThreadPool.this.getId();
可以看到同步范围仅限于this,也就是CachedThreadPool的实例,只有一个。但是static字段是类的字段,不是实例的字段,因此不在加锁范围!然而,同步会刷新代码块内所有用到的变量,不论static与否。而唯一实例使++代码块被单线程独占。两者结合,意外地做到了并发安全。
还可以试验一下,synchronized改成synchronized(CachedThreadPool.class){...},并把main标为synchronized,会导致死锁。
synchronized的知识:指定了一个同步范围,进出范围时会刷新相关变量,阻止其他线程进入该范围。synchronizedmethod的范围是this,synchronizedstaticmethod的范围是class。
补充:如果同一个类有的方法写了synchronized,有的方法没写,也达不到同步。
                            
查看完整回答
反对 回复 2019-04-19
  • 2 回答
  • 0 关注
  • 205 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信