JAVA多线程基石
^^^^^^^^^^^^^^
 - 作者:臭豆腐[trydofor.com]
 - 日期:2008-12-17
 - 授权:署名-非商业-保持一致 1.0 协议
 - 声明:拷贝、分发、呈现和表演本作品,请保留以上全部信息。

0. 文档目录
^^^^^^^^^^
[[<=$INDEX]]

1. 内存模型
^^^^^^^^^^^
主存储器(main memory),保存所有Object,为所有线程共有.
工作存储器(working memory),各线程私有空间.

线程无法直接读写主存储器,必须把必要数据复制(working copy)到工作存储器.
这些操作,由以下3组共6个原子操作(atomic action)组成.

 * lock/unlock 对实例进行获得/解除锁定
 * read/write  主存储器和工作存贮器的数据读/写
 * use/assign  工作存储器和线程的数据读/写
 
 .....................................................................
 ------------------------------------------------------------> 时间线
             ^ Object.field -----> use      |                  线程A
             |              <----  assign   |                  存储器A
 ---|--------|------------------------------|-------|--------
    |lock    |read                          |write  | unlock   操作
 ===v========|==============================|=======v========
             | Object.field                 v                  主存储器
                                                               (共享)
 ===================================================^========
    x can not lock but wait ...                     | lock   
 ---|-----------------------------------------------|--------                                                                
                                                               线程B 
                                                               存储器B
 .....................................................................

2. synchronized的作用
^^^^^^^^^^^^^^^^^^^^^
synchronized 是负责线程的共享互斥的.
被其保护的某个东西,需要进行线程同步和内存同步.

以下是常见的synchronized 写法.
======================== java : Synchronized.java ========================
/**
* methodSA 和methodSB 是等价的
*/
synchronized void methodSA(){
    // do something
}
void methodSB(){
    synchronized(this){
        // do something
    }
}

/**
* methodSC 和methodSD 是等价的
*/
static synchronized void methodSC(){
    // do something
}
static void methodSD(){
    synchronized(Synchronized.class){
        // do something
    }
}

=====================================================================

线程永远在自己的工作存储器上工作,当碰到synchronized 块时,会发生
  * 当线程将要进入时,完成2个动作.
    1) 把当前工作存储器的数据write到主存储器
    2) 把主存储器的数据read到工作存储器
  * 当线程将要离开时,完成1个动作.
    1) 把当前工作存储器的数据write到主存储器

3. volatile的作用
^^^^^^^^^^^^^^^^^
volatile 不负责线程的共享互斥,而是进行内存同步.

基本类型(primitive type),如char,int 的赋值和引用操作是原子级的.
对象等引用类型(reference type)的赋值和引用也是原子级的.
但是,long,double 的赋值和引用却不是原子级的.

当线程欲引用 volatile 字段的值时,通常会从主存储器拷贝到工作存储器,
相反,当线程赋值给 volatile字段后,通常会从工作存储器反映到主存储器.

4. Object的休息室
^^^^^^^^^^^^^^^
Object是java之源,也是synchronized和volatile保护的对象.
wait/notify/notifyAll是Object的方法,而不是Thread特有的.

休息室(wait set),是为了描述线程而引进的虚拟概念,它既不真实的也不可操作.
但是,所有实例都有这么一个休息室,是所有等待执行的线程的集合.暂停的线程,
永远会呆在休息室中,除非发生以下任一情况.

 * 有其他线程以 notify/notifyAll 方法唤醒该线程.
 * 有其他线程以 interrupt 方法唤醒该线程.
 * wait 方法已经到期.

因此 wait/notify/notifyAll,都是对休息室的操作以调度线程.