黑马程序员 – 学习Java5线程并发库(7) -- 2013.1.5
来源:互联网 发布:淘宝通栏轮播图片尺寸 编辑:程序博客网 时间:2024/05/04 20:34
----------- android培训、java培训、java学习型技术博客、期待与您交流! ------------
第三题:现有程序同时启动了4个线程去调用TestDo.doSome(key, value)方法,由于TestDo.doSome(key, value)方法内的代码是先暂停1秒,然后再输出以秒为单位的当前时间值,所以,会打印出4个相同的时间值,如下所示:
4:4:1258199615
1:1:1258199615
3:3:1258199615
1:2:1258199615
请修改代码,如果有几个线程调用TestDo.doSome(key, value)方法时,传递进去的key相等(equals比较为true),则这几个线程应互斥排队输出结果,即当有两个线程的key都是"1"时,它们中的一个要比另外其他线程晚1秒输出结果,如下所示:
4:4:1258199615
1:1:1258199615
3:3:1258199615
1:2:1258199616
总之,当每个线程中指定的key相等时,这些相等key的线程应每隔一秒依次输出时间值(要用互斥),如果key不同,则并行执行(相互之间不互斥)。原始代码如下:
package syn;
//不能改动此Test类
public class Test extends Thread {
private TestDo testDo;
private String key;
private String value;
public Test(String key,String key2,Stringvalue){
this.testDo = TestDo.getInstance();
/*常量"1"和"1"是同一个对象,下面这行代码就是要用"1"+""的方式产生新的对象,
以实现内容没有改变,仍然相等(都还为"1"),但对象却不再是同一个的效果*/
this.key = key+key2;
this.value = value;
}
public static void main(String[] args) throwsInterruptedException{
Test a = newTest("1","","1");
Test b = newTest("1","","2");
Test c = newTest("3","","3");
Test d = newTest("4","","4");
System.out.println("begin:"+(System.currentTimeMillis()/1000));
a.start();
b.start();
c.start();
d.start();
}
public void run(){
testDo.doSome(key, value);
}
}
class TestDo {
private TestDo() {}
private static TestDo _instance = newTestDo();
public static TestDo getInstance() {
return _instance;
}
public void doSome(Object key, Stringvalue) {
// 以大括号内的是需要局部同步的代码,不能改动!
{
try {
Thread.sleep(1000);
System.out.println(key+":"+value+ ":"
+(System.currentTimeMillis() / 1000));
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
分析:this.key = key+key2;这句话使得每个线程创建的key都不是同一个对象,所以不能直接拿key当做监视器对象。解决的想法是:把每一个传入进来的key当做当前线程的监视器对象,再把每一个key都存入一个集合中,当后面出现的key内容和以前的key相同时,则把监视器对象换为原来集合中已有的key,这样就能保证当key内容相同时,线程互斥,key不相同的线程并发执行。
由于有迭代器操作,还有往集合中添加元素的add操作,多线程时会出现安全问题。即在迭代的时候,另外一个线程有add操作,这时就会报异常。把集合换成新学习的CopyOnWriteArrayList就不会出现这个多线程安全问题了。
- 黑马程序员 – 学习Java5线程并发库(7) -- 2013.1.5
- 黑马程序员 – 学习Java5线程并发库(7) -- 2013.1.5
- 黑马程序员 – 学习Java5线程并发库(7) -- 2013.1.5
- 黑马程序员 – 学习Java5线程并发库(5) -- 2013.1.3
- 黑马程序员 – 学习Java5线程并发库(3) -- 2013.1.1
- 黑马程序员 – 学习Java5线程并发库(4) -- 2013.1.2
- 黑马程序员 – 学习Java5线程并发库(6) -- 2013.1.4
- 黑马程序员——Java5中的线程并发库(一)---概述、线程池、Callable和Future、Lock和Condition
- Java5 并发线程学习
- 黑马程序员:Java基础总结----线程池及java5的线程并发库
- java5线程并发库学习之Semaphore
- java5线程并发库学习之Semaphore
- java5线程并发库
- java5 线程并发库
- 黑马程序员——Java5中的线程并发库(二)--Semaphore实现信号灯、其他同步函数、可阻塞的队列、同步集合
- 黑马程序员---线程并发库
- 黑马程序员Java学习笔记之多线程(并发)
- (7)java5线程并发库的应用(线程池)
- shell脚本文件参数
- 如何避免被女人睡
- 叶匡政:为何要对腐败“零容忍”?
- 黑马程序员 – 学习Java5线程并发库(7) -- 2013.1.5
- 黑马程序员 – 学习Java5线程并发库(7) -- 2013.1.5
- 黑马程序员 – 学习Java5线程并发库(7) -- 2013.1.5
- oracle数据库导出表结构
- iPhone开发下载网络数据的几种方法总结
- creo2.0安装方法
- 黑马程序员 – 开始报名! -- 2013.1.16
- CC2530的IEEE地址
- 如何解决数据库中的字符型字段值中包含'0A'时,导出的文件用EXECEL打开时行数变多或者将结果导入数据库出错
- 跟我一起玩Win32开发(6):创建右键菜单
- 软件随想录:程序员部落酋长Joel谈软件(阮一峰译)-7