第一题
现有的程序代码模拟产生了16个日志对象,并且需要运行16秒才能打印完这些日志,请在程序中增加4个线程去调用parseLog()方法来分头打印这16个日志对象,程序只需要运行4秒即可打印完这些日志对象。
|
|
实现:通过阻塞队列实现线程间的通信
|
|
第二题
现成程序中的Test类中的代码在不断地产生数据,然后交给TestDo.doSome()方法去处理,就好像生产者在不断地产生数据,消费者在不断消费数据。
请将程序改造成有10个线程来消费生成者产生的数据,这些消费者都调用TestDo.doSome()方法去进行处理,故每个消费者都需要一秒才能处理完,程序应保证这些消费者线程依次有序地消费数据,只有上一个消费者消费完后,下一个消费者才能消费数据,下一个消费者是谁都可以,但要保证这些消费者线程拿到的数据是有顺序的。
|
|
在实现之前先介绍一个阻塞队列:SynchronousQuene,一种阻塞队列,其中每个插入操作必须等待另一个线程的对应移除操作 ,反之亦然。同步队列没有任何内部容量,甚至连一个队列的容量都没有。除非另一个线程试图移除某个元素,否则也不能(使用任何方法)插入元素;也不能迭代队列,因为其中没有元素可用于迭代。
应用:它非常适合于传递性设计,在这种设计中,在一个线程中运行的对象要将某些信息、事件或任务传递给在另一个线程中运行的对象,它就必须与该对象同步。
|
|
第三题
现有程序同时启动了4个线程去调用TestDo.doSome(key, value)方法,由于TestDo.doSome(key, value)方法内的代码是先暂停1秒,然后再输出以秒为单位的当前时间值,所以,会打印出4个相同的时间值,如下所示:
|
|
请修改代码,如果有几个线程调用TestDo.doSome(key, value)方法时,传递进去的key相等(equals比较为true),则这几个线程应互斥排队输出结果,即当有两个线程的key都是”1”时,它们中的一个要比另外其他线程晚1秒输出结果,如下所示:
|
|
总之,当每个线程中指定的key相等时,这些相等key的线程应每隔一秒依次输出时间值(要用互斥),如果key不同,则并行执行(相互之间不互斥)。
|
|
对于源代码中关于实现值相同而对象不同的效果进行解释:
对于:
编译器自动优化,所以a和b是同一个对象
而对于:key = key+key2; 由于是变量,编译器无法识别,这时a和b把“1”和“”赋值给key和key2时会得到两个不同的对象
思想:将集合中的对象作为同步代码块的锁,即this锁,每次将对象存入集合中的时候,就判断是否原集合中已经存在一个与将要存入集合的对象值相同的对象,即用equals比较,如果有,那么就获取原来的这个对象,把这个对象作为将要存入对象的锁,这样它们持有的就是同一把锁,即可实现互斥,这样就可以实现值相同的对象在不同的时刻打印的效果
代码中出现的问题:在遍历ArrayList集合查找与要存入值相同元素的时候,进行了添加的动作,所以会出现并发修改异常,因此使用并发的CopyOnWriteArrayList
|
|