- 浏览: 152409 次
- 性别:
- 来自: 大连
文章分类
最新评论
-
trydofor:
fujohnwang 写道为啥不是maven的问题?大概是因为 ...
jdk1.6.0_16 小bug -
fujohnwang:
为啥不是maven的问题?
jdk1.6.0_16 小bug -
trydofor:
参考资料http://supermmx.org/blog/20 ...
必须知道,别滥用-finally -
trydofor:
后续跟踪:使用javap分析finally块中return值h ...
必须知道,别滥用-finally -
trydofor:
(2)javap -c TestFor(1)javap -c ...
while(ture) 和 for(;;)
JAVA 多线程设计模式
^^^^^^^^^^^^^^^^^^^
- 作者:臭豆腐[trydofor.com]
- 日期:2008-12-24
- 授权:署名-非商业-保持一致 1.0 协议
- 声明:拷贝、分发、呈现和表演本作品,请保留以上全部信息。
0. 文档目录
^^^^^^^^^^
[[<=$INDEX]]
1. 适用范围与知识背景
^^^^^^^^^^^^^^^^^^^^^
本文是对"Java多线程设计模式",中国铁道出版社 (ISBN 7-113-06402-7)
一书的学习备忘,适用于java1.4.
实际项目中应该尽量回避多线程,如果必须使用,最好使用DougLee大师的杰作.
让我也具有DougLee一样能执行多线程的计算机头脑吧
2. Thread 状态图
^^^^^^^^^^^^^^^^
================= txt figure-1 : Thread Status =================
(_) Thread Status /jdk1.4
| new www.trydofor.com 2008-12-21
+------------+ =============================
| initial | * Object's method/status
+-----+------+ @ Thread's method/status
| @start() synchronized can not timeout
+------------+
| executable |<------------------------------------+
+-+----------+ @sleep() +----------+ @timeout |
| ^ +-------->|@sleeping +------------>|
| @yield | | +----------+ @interrupt()|
v | | |
+----------+-+ |*wait() +----------+ |
| running |--+-------->|*wait-set | @timeout |
+-----+------+ | +----+-----+ |
| | | *notify/All() |
| @run() | | @interrupt() |
| | synch- v |
v | ronized +----------+ acquire |
+------------+ +-------->|*race-set |------------>|
| final | | +----------+ |
+-----+------+ | +----------+ done |
| sys gc +-------->| blocking +------------>|
(8) block-io +----------+
================================================================
3. 脆弱独木桥(SingleThreadExecution)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
脆弱的独木桥,又称危险区(Critical Section),只能承重一个人,
因此一次只能允许一个线程执行.
当共享资源可以被多个线程访问,且资源状态会改变时,就可以考虑该模式.
对危险区的划分是重点,以人脑模拟电脑思考程序的执行过程.
同时要注意死锁和继承等对安全性的破坏.
如果看到 synchronized 相互嵌套,基本上就会死锁了.
以下代码和Thread.sleep()很像 :0
================== java :ThreadSleep.java ======================
public class ThreadSleep {
public static void sleep(long ms) throws InterruptedException{
if(ms>0){
Object obj = new Object();
synchronized(obj){
obj.wait(ms);
}
}
}
}
================================================================
4. 只许参观不许摸(Immutable)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^
像java.lang.String一样,实例一旦产生,状态就不再改变.
对于一个只读的东东,即便同时有无数个线程访问,也没什么问题.
final class/method/field/parameter 可以很大的提高安全性.
值得注意的是,不要出现以下代码中的问题,错把 mutable 数据暴露出去.
====================== java : Mutable.java =====================
public final class Mutable {
private final StringBuffer sb;
public Mutable(String str){
sb = new StringBuffer(str);
}
public StringBuffer getText(){
return sb;
}
}
================================================================
5. 不见不散(GuardedSuspension)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
如果条件不满足,就一直等,不见不散.
便于记忆,可以理解成多线程版的 if
====================== java : Queue.java =======================
import java.util.LinkedList;
public final class Queue {
private final LinkedList queue = new LinkedList();
public synchronized Object get(){ // 为什么同步this
while(queue.size()<=0){ // 为什么不用If
try{
wait(); // 为什么不是Thread.sleep()
}catch(InterruptedException e){
// can not be interrupted
}
}
return queue.removeFirst();
}
public synchronized void put(Object obj){
queue.addLast(obj);
notifyAll();
}
}
================================================================
6. 你们聊,我有事先走了(Balking)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
不见不散固然为人所敬仰,但不是所有时候都需要不见不散.比如,
* 不需要去执行时.
比如我经常神经性的Ctrl+s,但编辑器只有当文件变更时,才真正保持.
* 不想等到条件成立时.
小兔说:我妈妈叫我小兔兔.
小猪说:我妈妈叫我小猪猪.
小鸡说:你们聊,我有事先走了!
* 只能执行一次时.
发工资了,如果能多领几份该多好啊
把Balking的状态通知给调用者的方式.
* 忽略balk的发生.
* 以返回值表达,比如true/false
* 以异常的形式通知.
======================= java : Data.java =======================
public abstract class Data {
private boolean changed;
public final synchronized void change(){
changed = true;
}
public final synchronized void save(){
if(!changed) return;
doSomething();
changed = false;
}
protected abstract void doSomething();
}
================================================================
7. 生产者消费者(ProducerConsumer)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
生产者和消费者在不同的线程上运行,一般情况下,两者的处理速度存在很大差异.
因此在两者之间存在一个隐藏的"桥梁"来缓解这样的差异.管道(pipe)模式,是该
模式的简化形式.
所有参与者,
* Data 被传来传去的东西.
* Producer 生产Data的家伙.
* Customer 消灭Data的家伙.
* Channel 以上三者的纽带,是传送,调度的核心.
为什么Producer不能直接把Data给Customer呢?
因为这样的话,就是Producer线程来完成Customer的动作了.
参与者多了,有必要设置超时和可取消.好的习惯是让Chanel互斥的方法 throws
InterruptedException,隐式的告诉调用者,该方法具有以下含义,
* 我可能很耗时
* 我可以被取消
* 我可能调用了Object.wait(),Thread.sleep(),Thread.join().
注意,Object.wait()被interrupt的时候,需要重新获得锁,才会被执行.
====================== java : Channel.java =====================
import java.util.LinkedList;
public class Channel {
private final int capacity;
private final LinkedList datas;
public Channel(int capacity){
this.capacity = capacity;
this.datas = new LinkedList();
}
public synchronized void put(Object obj) throws InterruptedException{
while(datas.size()>=capacity){
wait();
}
datas.addLast(obj);
notifyAll();
}
public synchronized Object get() throws InterruptedException{
while(datas.size()<= 0){
wait();
}
Object obj = datas.removeFirst();
notifyAll();
return obj;
}
}
================================================================
8. 读写锁(ReadWriteLock)
^^^^^^^^^^^^^^^^^^^^^^^^
为了解决以下两种冲突,
* 读写(read-write conflict)
* 写写(write-write conflict)
在read和write的时候,都采用以下的结构,
===================== java : Example.java ======================
ReadWriteLock lock = new ReadWriteLock();
public String read() throws InterruptedException{
lock.readLock(); // 为什么不可以在try内呢?
try{
return doRead();
}finally{
lock.readUnlock();
}
}
public void write(String str) throws InterruptedException{
lock.writeLock();
try{
doWrite(str);
}finally{
lock.writeUnlock();
}
}
================================================================
================== java : ReadWriteLock.java ===================
public final class ReadWriteLock {
private int readingReaders = 0;
private int waitingWriters = 0; // 不计数会有什么问题?
private int writingWriters = 0;
private boolean preferWriter = true; // 为什么添加这个标记?
public synchronized void readLock() throws InterruptedException{
while(writingWriters>0 || (preferWriter && waitingWriters>0)){
wait();
}
readingReaders++;
}
public synchronized void readUnlock() throws InterruptedException{
readingReaders --;
preferWriter = true;
notifyAll(); // 可能唤醒哪些线程(Reader/Writer)?
}
public synchronized void writeLock() throws InterruptedException{
waitingWriters ++;
try{
while(readingReaders>0|| writingWriters>0){
wait();
}
}finally{
waitingWriters --;
}
writingWriters ++;
}
public synchronized void writeUnlock() throws InterruptedException{
writingWriters --;
preferWriter = false;
notifyAll(); // 可能唤醒哪些线程(Reader/Writer)?
}
}
================================================================
9. 临时工/合同工(ThreadPerMessage/WokerThread)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
这两种模式的基本特点,就是把工作交给其他的线程去处理,
把调用(invocation)和执行(execution)放在不同个线程中,以提高响应能力.
临时工相当灵活,呼之即来,挥之即去,不过缺点是浪费资源,不便控制.
合同工则可以弥补上述缺点,让工作持续稳定.
一般来讲,多使用线程池技术,和既定的安全框架,不多累述.来道头脑风暴吧,
====================== java : Test.java ========================
public class Test {
public void doOther(final Object obj){
new Thread(){
public void run(){
System.out.println(obj);
// Business Method
}
}.start();
}
}
================================================================
完成method方法,使下面程序只输出"12",而不输出"3";
不可以在method中print任何字符出来.
====================== java : Quest.java =======================
public class Quest {
public static void haha(Object obj){
System.out.print("1");
method(obj);
System.out.print("2");
synchronized(obj){
System.out.print("3");
}
}
public static void main(String[] args) {
haha("");
}
}
================================================================
一种答案:
==================== aes : 密码是trydofor ======================
WXDDkl4eASEwISEwIS0fccK1w6jCkCgoBcOVwpEEWmvCjBFDLRrDiwE4w7/CpSEx
NjAhZMO2w4XDicK1w4k9w4tELQ7CnyExMyFsw6dsL1Qfw7RRwqjDhcK4w49/LRjD
oRdCw7hmw47ChkPCjcODw73CqijDksKvLXIIW0TDqnbDvXbCisOhw7bCkFxrR8Ks
LS9dwpTDhgg1c8KLYyExMiHCiCkWwrs7wpotITEzIcOyw608wpd2Lj/Dt1owecOo
ITExIQPDhC3CihBpwrXCrCEzNCHCjnsSccOuwovCq8KlN8KCLXUvwpM6a3LDimXC
vWVmUsOfPDPCpC3CqsONSMKRw5N8IMKtcsKecELDq8OgwoZOLV8dw4zDpDEPwo3C
gMKkw50fVGghMTEhcjUtw455KHIdICMhMTIhfMK2ccOEccOFSMK7LRrDgcOIM1Yh
MzkhwoDDgxlpwoXCtMKww6Q3wqgtwpVzM8KSwooqw7LCv1TDi0hwc8O2wqw7LSbD
hUIRw7hxw6nDpkDDssOBYx/Dt8KQwoQtaMOfX8K6w6vDpMO/SW0Eb0fCpsKzITM0
IcKyLTDDlE3DvnbDjC7Cl8KAw7wVdiExMSE4wqN5LcOcBR7DnMO6wqlwYsOaEsKK
PhPDlMKQGC0GITE2MCHDpMK+KsOzMlvClBDDimgRFz8GLcOLHcONwrYpw7gjw7lM
V8KLwpp4ITkhVcOzLcKGQ8ODKcKSw4HDpMO2H34pF8O5GcKkOi1+VmMYwrjDrBPD
rmATGUBjFsKKwpYtITM5ISjDusOBw6fDtTDCrXXDgMKnwosBE1tyLU0RRMOvwp1M
wr3Drj/Cnz5UaHtEUi0FWCrDmTVYw6nChMKmQMKnw7NiLEXCgS0hMTMhMMOvO28r
Z8O9w4PCsHTCkA4vXEEtM8KLw71nWBJINDLDrS4PwrjDjzzDuC3CkcK/OMKMwpXC
vcOzw5R3HcKzUX7CrE8uLcKEYxMGb8Kpw7dLITAhBQTCqsKVwpfDjHUtwoMSccKf
wo5hw6cyw70kw4pawocqwoXDrS3Dlm/CoUU4wqtXBlQVwr5hD8KTcEAtf8KmOzIh
MTMhwr3DpcKJOlfDqcK1E8KNwqPCpC3CmsKKwrzCmBZEwpIBw59TbyBZwosZw60t
X8O0wo0GwpvDozVMJX0eFMOiFm9zLcKnwp7CrkpVwo3DlMK/dcKdSijDhMOJw5oP
LQjDrhHDl8Knw58hMTAhw7Y9Mx/DtMO0wrbDjAYtwrBcw6LDiCVtworDmsK0w6fD
iMKbwqrCgHgBLcODwoTDkCExMyHDkMKnEMKTw4ojMErCjBUwwpItUsKNwr/CucOe
N1HCn1jCm8OsBmnCgwVrLcOlwqN6LmnDksKAY01vPSEzMyHCmiEwIWtMLcKhw5dU
wprCiCEzNCHDlEJLw6XDisO0wqzDvh7Diy3CvMKFITEzITM6w5rCvMOMITEwIcOK
wpjDuAM=
================================================================
10. 拿订单提货(Future)
^^^^^^^^^^^^^^^^^^^^^
要那么长时间才完成啊,我先占个座吧,好了告诉我.
Future模式,可以设置返回值,也可以采用callback,这依赖于订货者的要求.
==================== java : FutureData.java ====================
public class FutureData implements Data {
private Object data = null; // 真正返回值
private volatile boolean ready = false;
public synchronized void set(Object obj){
if(ready){
return;
}
data = obj;
ready = true;
notifyAll();
}
public synchronized Object get(){ // 一直等下去
while(!ready){
try {
wait();
} catch (InterruptedException e) {
}
}
return data;
}
public boolean isReady(){ // 可以来检查.
return ready;
}
}
================================================================
11. 你还是自裁吧(TwoPhaseTermination)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
从外界终止一个执行中的线程是不安全的,最好的办法是让它自己搞定.
不要使用Thread.stop(),它已经被 deprecated 的了,因为他强制终止线程.
==================== java : DoItYouself.java ===================
public class DoItYouself extends Thread{
private volatile boolean needShutdown = false; // 为何 volatile
public void shutdown(){
needShutdown = true;
interrupt(); // 为什么呢?
}
public final void run(){
try{
while(!needShutdown){
//doWork();
}
}catch(InterruptedException e){
}finally{
//doShutdown();
}
}
}
================================================================
另外,下面代码会输出"13",因为ShutdownHook Thread.start()会在System.exit()或
所有非Deamon线程结束时被调用.
==================== java : DoItYouself.java ===================
public class ShutdownHook{
public static void main(String[] args) {
Runtime.getRuntime().addShutdownHook(
new Thread(){
public void run(){
System.out.print("3");
}
}
);
System.out.print("1");
System.exit(0);
System.out.print("2");
}
}
================================================================
[[!拓展内容, interrupt() 的作用.]]
调用interrupt(),会产生以下作用之一,不可兼得中断异常和中断状态.
1) 当线程sleep/wait/join时,抛出中断异常(InterruptedException).
2) 非1)时,线程变成中断状态(interrupted),可以使用Thread.interrupted();
======================== java : 相互转换 =======================
InterruptedException se = null;
try{
if(Thread.interrupted()){
throw new InterruptedException(); // 状态 => 异常
}
}catch(InterruptedException e){
Thread.currentThread().interrupt(); // 异常 => 状态
se = e; // 异常 => 异常
}finally{
//doShutdown();
}
if(se != null) throw e;
================================================================
12. 私有保险箱(ThreadLocalStorage)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
ThreadLocal模式下的多线程行为表面上看与单线程很像,可以不用考虑互斥.
这种以线程为主键的map,可以很好的分离开线程私有的信息,就像局部变量一样.
------------
排版格式:http://www.trydofor.com/a9w3-auhome/trydofor/article/2008/1224135056/body.htm
^^^^^^^^^^^^^^^^^^^
- 作者:臭豆腐[trydofor.com]
- 日期:2008-12-24
- 授权:署名-非商业-保持一致 1.0 协议
- 声明:拷贝、分发、呈现和表演本作品,请保留以上全部信息。
0. 文档目录
^^^^^^^^^^
[[<=$INDEX]]
1. 适用范围与知识背景
^^^^^^^^^^^^^^^^^^^^^
本文是对"Java多线程设计模式",中国铁道出版社 (ISBN 7-113-06402-7)
一书的学习备忘,适用于java1.4.
实际项目中应该尽量回避多线程,如果必须使用,最好使用DougLee大师的杰作.
让我也具有DougLee一样能执行多线程的计算机头脑吧
2. Thread 状态图
^^^^^^^^^^^^^^^^
================= txt figure-1 : Thread Status =================
(_) Thread Status /jdk1.4
| new www.trydofor.com 2008-12-21
+------------+ =============================
| initial | * Object's method/status
+-----+------+ @ Thread's method/status
| @start() synchronized can not timeout
+------------+
| executable |<------------------------------------+
+-+----------+ @sleep() +----------+ @timeout |
| ^ +-------->|@sleeping +------------>|
| @yield | | +----------+ @interrupt()|
v | | |
+----------+-+ |*wait() +----------+ |
| running |--+-------->|*wait-set | @timeout |
+-----+------+ | +----+-----+ |
| | | *notify/All() |
| @run() | | @interrupt() |
| | synch- v |
v | ronized +----------+ acquire |
+------------+ +-------->|*race-set |------------>|
| final | | +----------+ |
+-----+------+ | +----------+ done |
| sys gc +-------->| blocking +------------>|
(8) block-io +----------+
================================================================
3. 脆弱独木桥(SingleThreadExecution)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
脆弱的独木桥,又称危险区(Critical Section),只能承重一个人,
因此一次只能允许一个线程执行.
当共享资源可以被多个线程访问,且资源状态会改变时,就可以考虑该模式.
对危险区的划分是重点,以人脑模拟电脑思考程序的执行过程.
同时要注意死锁和继承等对安全性的破坏.
如果看到 synchronized 相互嵌套,基本上就会死锁了.
以下代码和Thread.sleep()很像 :0
================== java :ThreadSleep.java ======================
public class ThreadSleep {
public static void sleep(long ms) throws InterruptedException{
if(ms>0){
Object obj = new Object();
synchronized(obj){
obj.wait(ms);
}
}
}
}
================================================================
4. 只许参观不许摸(Immutable)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^
像java.lang.String一样,实例一旦产生,状态就不再改变.
对于一个只读的东东,即便同时有无数个线程访问,也没什么问题.
final class/method/field/parameter 可以很大的提高安全性.
值得注意的是,不要出现以下代码中的问题,错把 mutable 数据暴露出去.
====================== java : Mutable.java =====================
public final class Mutable {
private final StringBuffer sb;
public Mutable(String str){
sb = new StringBuffer(str);
}
public StringBuffer getText(){
return sb;
}
}
================================================================
5. 不见不散(GuardedSuspension)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
如果条件不满足,就一直等,不见不散.
便于记忆,可以理解成多线程版的 if
====================== java : Queue.java =======================
import java.util.LinkedList;
public final class Queue {
private final LinkedList queue = new LinkedList();
public synchronized Object get(){ // 为什么同步this
while(queue.size()<=0){ // 为什么不用If
try{
wait(); // 为什么不是Thread.sleep()
}catch(InterruptedException e){
// can not be interrupted
}
}
return queue.removeFirst();
}
public synchronized void put(Object obj){
queue.addLast(obj);
notifyAll();
}
}
================================================================
6. 你们聊,我有事先走了(Balking)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
不见不散固然为人所敬仰,但不是所有时候都需要不见不散.比如,
* 不需要去执行时.
比如我经常神经性的Ctrl+s,但编辑器只有当文件变更时,才真正保持.
* 不想等到条件成立时.
小兔说:我妈妈叫我小兔兔.
小猪说:我妈妈叫我小猪猪.
小鸡说:你们聊,我有事先走了!
* 只能执行一次时.
发工资了,如果能多领几份该多好啊
把Balking的状态通知给调用者的方式.
* 忽略balk的发生.
* 以返回值表达,比如true/false
* 以异常的形式通知.
======================= java : Data.java =======================
public abstract class Data {
private boolean changed;
public final synchronized void change(){
changed = true;
}
public final synchronized void save(){
if(!changed) return;
doSomething();
changed = false;
}
protected abstract void doSomething();
}
================================================================
7. 生产者消费者(ProducerConsumer)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
生产者和消费者在不同的线程上运行,一般情况下,两者的处理速度存在很大差异.
因此在两者之间存在一个隐藏的"桥梁"来缓解这样的差异.管道(pipe)模式,是该
模式的简化形式.
所有参与者,
* Data 被传来传去的东西.
* Producer 生产Data的家伙.
* Customer 消灭Data的家伙.
* Channel 以上三者的纽带,是传送,调度的核心.
为什么Producer不能直接把Data给Customer呢?
因为这样的话,就是Producer线程来完成Customer的动作了.
参与者多了,有必要设置超时和可取消.好的习惯是让Chanel互斥的方法 throws
InterruptedException,隐式的告诉调用者,该方法具有以下含义,
* 我可能很耗时
* 我可以被取消
* 我可能调用了Object.wait(),Thread.sleep(),Thread.join().
注意,Object.wait()被interrupt的时候,需要重新获得锁,才会被执行.
====================== java : Channel.java =====================
import java.util.LinkedList;
public class Channel {
private final int capacity;
private final LinkedList datas;
public Channel(int capacity){
this.capacity = capacity;
this.datas = new LinkedList();
}
public synchronized void put(Object obj) throws InterruptedException{
while(datas.size()>=capacity){
wait();
}
datas.addLast(obj);
notifyAll();
}
public synchronized Object get() throws InterruptedException{
while(datas.size()<= 0){
wait();
}
Object obj = datas.removeFirst();
notifyAll();
return obj;
}
}
================================================================
8. 读写锁(ReadWriteLock)
^^^^^^^^^^^^^^^^^^^^^^^^
为了解决以下两种冲突,
* 读写(read-write conflict)
* 写写(write-write conflict)
在read和write的时候,都采用以下的结构,
===================== java : Example.java ======================
ReadWriteLock lock = new ReadWriteLock();
public String read() throws InterruptedException{
lock.readLock(); // 为什么不可以在try内呢?
try{
return doRead();
}finally{
lock.readUnlock();
}
}
public void write(String str) throws InterruptedException{
lock.writeLock();
try{
doWrite(str);
}finally{
lock.writeUnlock();
}
}
================================================================
================== java : ReadWriteLock.java ===================
public final class ReadWriteLock {
private int readingReaders = 0;
private int waitingWriters = 0; // 不计数会有什么问题?
private int writingWriters = 0;
private boolean preferWriter = true; // 为什么添加这个标记?
public synchronized void readLock() throws InterruptedException{
while(writingWriters>0 || (preferWriter && waitingWriters>0)){
wait();
}
readingReaders++;
}
public synchronized void readUnlock() throws InterruptedException{
readingReaders --;
preferWriter = true;
notifyAll(); // 可能唤醒哪些线程(Reader/Writer)?
}
public synchronized void writeLock() throws InterruptedException{
waitingWriters ++;
try{
while(readingReaders>0|| writingWriters>0){
wait();
}
}finally{
waitingWriters --;
}
writingWriters ++;
}
public synchronized void writeUnlock() throws InterruptedException{
writingWriters --;
preferWriter = false;
notifyAll(); // 可能唤醒哪些线程(Reader/Writer)?
}
}
================================================================
9. 临时工/合同工(ThreadPerMessage/WokerThread)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
这两种模式的基本特点,就是把工作交给其他的线程去处理,
把调用(invocation)和执行(execution)放在不同个线程中,以提高响应能力.
临时工相当灵活,呼之即来,挥之即去,不过缺点是浪费资源,不便控制.
合同工则可以弥补上述缺点,让工作持续稳定.
一般来讲,多使用线程池技术,和既定的安全框架,不多累述.来道头脑风暴吧,
====================== java : Test.java ========================
public class Test {
public void doOther(final Object obj){
new Thread(){
public void run(){
System.out.println(obj);
// Business Method
}
}.start();
}
}
================================================================
完成method方法,使下面程序只输出"12",而不输出"3";
不可以在method中print任何字符出来.
====================== java : Quest.java =======================
public class Quest {
public static void haha(Object obj){
System.out.print("1");
method(obj);
System.out.print("2");
synchronized(obj){
System.out.print("3");
}
}
public static void main(String[] args) {
haha("");
}
}
================================================================
一种答案:
==================== aes : 密码是trydofor ======================
WXDDkl4eASEwISEwIS0fccK1w6jCkCgoBcOVwpEEWmvCjBFDLRrDiwE4w7/CpSEx
NjAhZMO2w4XDicK1w4k9w4tELQ7CnyExMyFsw6dsL1Qfw7RRwqjDhcK4w49/LRjD
oRdCw7hmw47ChkPCjcODw73CqijDksKvLXIIW0TDqnbDvXbCisOhw7bCkFxrR8Ks
LS9dwpTDhgg1c8KLYyExMiHCiCkWwrs7wpotITEzIcOyw608wpd2Lj/Dt1owecOo
ITExIQPDhC3CihBpwrXCrCEzNCHCjnsSccOuwovCq8KlN8KCLXUvwpM6a3LDimXC
vWVmUsOfPDPCpC3CqsONSMKRw5N8IMKtcsKecELDq8OgwoZOLV8dw4zDpDEPwo3C
gMKkw50fVGghMTEhcjUtw455KHIdICMhMTIhfMK2ccOEccOFSMK7LRrDgcOIM1Yh
MzkhwoDDgxlpwoXCtMKww6Q3wqgtwpVzM8KSwooqw7LCv1TDi0hwc8O2wqw7LSbD
hUIRw7hxw6nDpkDDssOBYx/Dt8KQwoQtaMOfX8K6w6vDpMO/SW0Eb0fCpsKzITM0
IcKyLTDDlE3DvnbDjC7Cl8KAw7wVdiExMSE4wqN5LcOcBR7DnMO6wqlwYsOaEsKK
PhPDlMKQGC0GITE2MCHDpMK+KsOzMlvClBDDimgRFz8GLcOLHcONwrYpw7gjw7lM
V8KLwpp4ITkhVcOzLcKGQ8ODKcKSw4HDpMO2H34pF8O5GcKkOi1+VmMYwrjDrBPD
rmATGUBjFsKKwpYtITM5ISjDusOBw6fDtTDCrXXDgMKnwosBE1tyLU0RRMOvwp1M
wr3Drj/Cnz5UaHtEUi0FWCrDmTVYw6nChMKmQMKnw7NiLEXCgS0hMTMhMMOvO28r
Z8O9w4PCsHTCkA4vXEEtM8KLw71nWBJINDLDrS4PwrjDjzzDuC3CkcK/OMKMwpXC
vcOzw5R3HcKzUX7CrE8uLcKEYxMGb8Kpw7dLITAhBQTCqsKVwpfDjHUtwoMSccKf
wo5hw6cyw70kw4pawocqwoXDrS3Dlm/CoUU4wqtXBlQVwr5hD8KTcEAtf8KmOzIh
MTMhwr3DpcKJOlfDqcK1E8KNwqPCpC3CmsKKwrzCmBZEwpIBw59TbyBZwosZw60t
X8O0wo0GwpvDozVMJX0eFMOiFm9zLcKnwp7CrkpVwo3DlMK/dcKdSijDhMOJw5oP
LQjDrhHDl8Knw58hMTAhw7Y9Mx/DtMO0wrbDjAYtwrBcw6LDiCVtworDmsK0w6fD
iMKbwqrCgHgBLcODwoTDkCExMyHDkMKnEMKTw4ojMErCjBUwwpItUsKNwr/CucOe
N1HCn1jCm8OsBmnCgwVrLcOlwqN6LmnDksKAY01vPSEzMyHCmiEwIWtMLcKhw5dU
wprCiCEzNCHDlEJLw6XDisO0wqzDvh7Diy3CvMKFITEzITM6w5rCvMOMITEwIcOK
wpjDuAM=
================================================================
10. 拿订单提货(Future)
^^^^^^^^^^^^^^^^^^^^^
要那么长时间才完成啊,我先占个座吧,好了告诉我.
Future模式,可以设置返回值,也可以采用callback,这依赖于订货者的要求.
==================== java : FutureData.java ====================
public class FutureData implements Data {
private Object data = null; // 真正返回值
private volatile boolean ready = false;
public synchronized void set(Object obj){
if(ready){
return;
}
data = obj;
ready = true;
notifyAll();
}
public synchronized Object get(){ // 一直等下去
while(!ready){
try {
wait();
} catch (InterruptedException e) {
}
}
return data;
}
public boolean isReady(){ // 可以来检查.
return ready;
}
}
================================================================
11. 你还是自裁吧(TwoPhaseTermination)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
从外界终止一个执行中的线程是不安全的,最好的办法是让它自己搞定.
不要使用Thread.stop(),它已经被 deprecated 的了,因为他强制终止线程.
==================== java : DoItYouself.java ===================
public class DoItYouself extends Thread{
private volatile boolean needShutdown = false; // 为何 volatile
public void shutdown(){
needShutdown = true;
interrupt(); // 为什么呢?
}
public final void run(){
try{
while(!needShutdown){
//doWork();
}
}catch(InterruptedException e){
}finally{
//doShutdown();
}
}
}
================================================================
另外,下面代码会输出"13",因为ShutdownHook Thread.start()会在System.exit()或
所有非Deamon线程结束时被调用.
==================== java : DoItYouself.java ===================
public class ShutdownHook{
public static void main(String[] args) {
Runtime.getRuntime().addShutdownHook(
new Thread(){
public void run(){
System.out.print("3");
}
}
);
System.out.print("1");
System.exit(0);
System.out.print("2");
}
}
================================================================
[[!拓展内容, interrupt() 的作用.]]
调用interrupt(),会产生以下作用之一,不可兼得中断异常和中断状态.
1) 当线程sleep/wait/join时,抛出中断异常(InterruptedException).
2) 非1)时,线程变成中断状态(interrupted),可以使用Thread.interrupted();
======================== java : 相互转换 =======================
InterruptedException se = null;
try{
if(Thread.interrupted()){
throw new InterruptedException(); // 状态 => 异常
}
}catch(InterruptedException e){
Thread.currentThread().interrupt(); // 异常 => 状态
se = e; // 异常 => 异常
}finally{
//doShutdown();
}
if(se != null) throw e;
================================================================
12. 私有保险箱(ThreadLocalStorage)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
ThreadLocal模式下的多线程行为表面上看与单线程很像,可以不用考虑互斥.
这种以线程为主键的map,可以很好的分离开线程私有的信息,就像局部变量一样.
------------
排版格式:http://www.trydofor.com/a9w3-auhome/trydofor/article/2008/1224135056/body.htm
评论
2 楼
trydofor
2009-01-04
跑到这个版来了,找了半天.
以后不发基础的东西了,JE都是architecture的.
以后不发基础的东西了,JE都是architecture的.
1 楼
flashgoods
2008-12-29
]呵呵,大家现在似乎都不活跃了。加油啊,一起来盖啊。
发表评论
-
Java 大括号语法糖
2012-07-04 16:53 1900Java 大括号语法糖 Java 的大括号(‘{’ ,curl ... -
jdk1.6.0_16 小bug
2011-06-01 17:10 2283汉语错误信息: [ERROR] \a9zones\a9id- ... -
JSR305 有助于提高代码健壮性
2011-05-18 16:44 2148JSR305 有助于提高代码健壮性 下载(java5+ ,据 ... -
Java常用工具类 -- ALOE 芦荟工程
2011-05-10 15:32 1349最牛全角互转半角 ... -
不可 T extends Iterator<E> & Iterable<E>
2011-05-09 10:50 1301//(a) public <T extends I ... -
必须知道,别滥用-finally
2011-04-21 10:57 1420public class Test { ... -
while(ture) 和 for(;;)
2011-03-28 11:10 1202多线程下的if,经常是while(...) 对于死等,在jdk ... -
小细节导致假调用
2011-03-17 11:40 896private final A9Loader< ... -
多线程情况下的一点差异
2011-03-14 17:04 836以下代码,没实际意义。 注意以下情况的不同点。 1)volat ... -
Number 为何不 Comparable
2011-02-18 09:55 1204现状 java 5+ package java.lang. ... -
synchronized 处发生了什么
2011-01-13 14:45 786问题 1.为何 map.clear(); 时的输出的毫秒数不连 ... -
讨论下 A9ID/A9IC 的短板
2011-01-12 16:21 902ID 生成策略,是各有各的道。且不问是否分布,多进程,多线程。 ... -
A9ID/A9IC
2011-01-12 09:35 1064第一版的A9ID 和A9IC已经出炉了. 定义如下 ... -
文字替换之效率
2010-11-02 10:40 11471. 常用方法 a) public String repl ... -
遍历 List 之效率
2010-11-02 09:30 13051. 常用遍历 List 之方法 遍历 List 方法 ... -
JAVA乱码你知道(DIY)
2010-11-01 13:45 10151. 四个涉及编码的环节 java开发中的乱码问题,有 ... -
超级基础:作用范围和生命周期
2010-10-30 14:29 780代码中的类或变量: 1。一旦创建了,就不在需要了。 ... -
码工考古之DataSync
2010-06-19 20:50 1056码工考古之DataSync 作者:臭豆腐[trydo ... -
IVY+IVYDE 非经典实践
2010-06-07 22:15 1098http://trydofor.com/a9w3-auhome ... -
日文编码SHIFT_JIS/MS932使用
2010-04-16 13:38 4132总结:在多数情况下,使用MS932代替SHIFT_JIS,可减 ...
相关推荐
Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式上传文件Java多线程设计模式...
Java多线程设计模式是Java开发中的重要领域,它涉及到如何在并发环境下高效、安全地管理资源和控制程序执行流程。本资料集包含了清晰完整的PDF版书籍和源代码,为学习和理解Java多线程设计模式提供了丰富的素材。 ...
标题和描述均指向了一个关于Java多线程设计模式的PDF文档的下载链接,这暗示了文档的主要内容将围绕Java中的多线程编程及其设计模式展开。在Java领域,多线程是一个核心概念,它允许程序执行多个任务同时进行,极大...
书中包含Java线程的介绍导读、12个重要的线程设计模式和全书总结以及丰富的附录内容。第一章相关线程设计模式的介绍,都举一反三使读者学习更有效。最后附上练习问题,让读者可以温故而知新,能快速地吸收书中的...
标题“java多线程设计模式详解.pdf”中提到的知识点是关于Java多线程编程中设计模式的应用。Java多线程是并发编程的重要组成部分,设计模式则是软件工程中用于解决特定问题的最佳实践。将两者结合起来,意味着此文件...
Java多线程设计模式是Java开发中的重要领域,它涉及到如何高效、安全地利用系统资源进行并发处理。在这个主题中,我们将深入探讨单线程、生产者与消费者模型以及Java中实现多线程的各种方法。 首先,单线程是程序...
### Java多线程设计模式详解 #### 一、Java多线程基础 Java语言自诞生以来,便内置了对多线程的支持。多线程可以让应用程序在同一时间处理多个任务,提高程序的执行效率和响应速度。Java中创建线程有两种基本方式...
本书《JAVA多线程设计模式》针对Java语言的多线程编程进行深入讲解,采用易于理解的方式介绍了与Java线程相关的多个设计模式,并通过实例程序与UML图示辅助阐述。书中的关键代码片段都有标注,易于读者理解与学习,...
java多线程设计模式,作者是:结城 浩,由 博硕文化 译。2005年4月,由中国铁道出版社出版。内附带部分源代码。
JAVA多线程设计模式_中国铁道出版社 本书浅显易懂的介绍了JAVA线程相关的设计模式,通过程序范例和UML图示来一一解说,书中代码的重要部分加了标注以使读者更加容易理解,再加上图文并茂,对于初学者还是程序设计...
Java 多线程 设计模式
Java多线程设计模式是Java并发编程中的一种高级技巧,它可以帮助开发者在处理并发问题时,提高代码的可读性、可维护性和性能。多线程设计模式是基于Java的并发API,如Thread、Runnable、ExecutorService等,通过特定...