15.线程面试题三

现有程序同时启动了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不同,则并行执行(相互之间不互斥)。原始代码如下:

//不能改动此Test类	
public class Test extends Thread {
	private TestDo testDo;
	private String key;
	private String value;

	public Test(String key, String key2, String value) {
		this.testDo = TestDo.getInstance();
		/*
		 * 常量"1"和"1"是同一个对象,下面这行代码就是要用"1"+""的方式产生新的对象,
		 * 以实现内容没有改变,仍然相等(都还为"1"),但对象却不再是同一个的效果
		 */
		this.key = key + key2;
		this.value = value;
	}

	public static void main(String[] args) throws InterruptedException {
		Test a = new Test("1", "", "1");
		Test b = new Test("1", "", "2");
		Test c = new Test("3", "", "3");
		Test d = new Test("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 = new TestDo();
	public static TestDo getInstance() {
		return _instance;
	}
	public void doSome(Object key, String value) {
		// 以大括号内的是需要局部同步的代码,不能改动!
		{
			try {
				Thread.sleep(1000);
				System.out.println(key + ":" + value + ":"
						+ (System.currentTimeMillis() / 1000));
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}
}
改进后的代码:

import java.util.Iterator;
import java.util.concurrent.CopyOnWriteArrayList;
//不能改动此Test类	
public class Test extends Thread {
	private TestDo testDo;
	private String key;
	private String value;

	public Test(String key, String key2, String value) {
		this.testDo = TestDo.getInstance();
		/*
		 * 常量"1"和"1"是同一个对象,下面这行代码就是要用"1"+""的方式产生新的对象,
		 * 以实现内容没有改变,仍然相等(都还为"1"),但对象却不再是同一个的效果
		 */
		this.key = key + key2;
		/**
		 *  a = "1" + "";
		 *  b = "1" + "";
		 *  此时a,b是同一个对象,因为程序还没有在运行的时候,编译器自动地
		 *  去优化成a = "1",b = "1",所以是同一个对象,而上面不是同一个对象,
		 *  因为是变量不是常亮
		 */
		this.value = value;
	}

	public static void main(String[] args) throws InterruptedException {
		Test a = new Test("1", "", "1");
		Test b = new Test("1", "", "2");
		Test c = new Test("3", "", "3");
		Test d = new Test("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 = new TestDo();
	public static TestDo getInstance() {
		return _instance;
	}
	/**
	 * 用private ArrayList keys = new ArrayList();会产生错误,如下
	 * Exception in thread "Thread-1" java.util.ConcurrentModificationException
		at java.util.ArrayList$Itr.checkForComodification(Unknown Source)
		at java.util.ArrayList$Itr.next(Unknown Source)
		at syn.TestDo.doSome(Test.java:60)
		at syn.Test.run(Test.java:42)
	 */
	private CopyOnWriteArrayList keys = new CopyOnWriteArrayList();
	public void doSome(Object key, String value) {
		Object o = key;
		if(!keys.contains(o)){
			keys.add(o);
		}else{
			for(Iterator iter = keys.iterator();iter.hasNext();){
				Object oo = iter.next();
				if(oo.equals(o)){
					o = oo;
					break;
				}
			}
		}
		synchronized(o)
		// 以大括号内的是需要局部同步的代码,不能改动!
		{
			try {
				Thread.sleep(1000);
				System.out.println(key + ":" + value + ":"
						+ (System.currentTimeMillis() / 1000));
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}
}

结果:

begin:1491226685
1:1:1491226686
3:3:1491226686
4:4:1491226686
1:2:1491226687

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值