synchronized用法

来源:互联网 发布:Ubuntu centos 编辑:程序博客网 时间:2024/05/29 16:43
假设P1、P2是同一个类的不同对象,这个类中定义了以下几种情况的同步块或同步方法,P1、P2就都可以调用它们。

1. 把synchronized当作函数修饰符(获取的是对象锁)
例如:
public synchronized void methodAAA() {
    //方法体
}
它锁定的是调用这个同步方法对象。也就是说,当一个对象P1在不同的线程中执行这个同步方法时,它们之间会形成互斥,达到同步的效果。但是这个对象所属的Class所产生的另一对象P2却可以任意调用这个被加了synchronized关键字的方法。
等同如下代码:
public void methodAAA() {
       synchronized ( this){//this指的是调用这个方法的对象
       //方法体
       }
}
同步方法实质是将synchronized作用于object reference。那个拿到了P1对象锁的线程,才可以调用P1的同步方法,而对P2而言,P1这个锁与它毫不相干,程序也可能在这种情形下摆脱同步机制的控制,造成数据混乱。

2.同步块(获取的是对象锁)
public void method3(SomeObject so ) {
            synchronized ( so) {
                 // …..
           }
}
这时,锁就是so这个对象,谁拿到这个锁谁就可以运行它所控制的那段代码。当有一个明确的对象作为锁时,就可以这样写程序,但当没有明确的对象作为锁,只是想让一段代码同步时,可以创建一个特殊的instance变量(它得是一个对象)来充当锁:
class Foo implements Runnable{
         private byte[] lock = new byte[0];  // 特殊的instance变量
         Public void methodA(){
            synchronized(lock) { //… }
         }
         //…..
}
注:零长度的byte数组对象创建起来将比任何对象都经济――查看编译后的字节码:生成零长度的byte[]对象只需3条操作码,而Object lock = new Object()则需要7行操作码。

3.将synchronized作用于static 函数(获取的是类锁)
Class Foo{
public synchronized static void methodAAA()   // 同步的static 函数
{
//….
}
public void methodBBB(){
       synchronized(Foo.class)   //  class literal(类名称字面常量)
}
}
代码中的methodBBB()方法是把class literal作为锁的情况,它和同步的static函数产生的效果是一样的,取得的锁很特别,是当前调用这个方法的对象所属的类(Class,而不再是由这个Class产生的某个具体对象了)。