synchronized关键字

643

锁一个对象

synchronized关键字用于处理多线程的竞争条件,被synchronized关键字修饰的对象synchronized(obj)就相当于给obj对象上了一把锁,谁拿到了锁谁就可以执行synchronized代码块下的代码,这种方法更像是把锁住的对象实例作为一个信号量,通过这个信号量判断当前线程有没有资格执行代码块中的东西,除此之外锁的作用范围跟这个obj对象没什么关系。

public Object obj;

public void fun(){
    obj.funA();
    synchronized(obj){
       obj.funB();
    }
}

根据上面结论,给obj对象上了锁只会影响synchronized代码块中的内容,同一时间只有一个线程可以执行代码块中的代码,而obj.funA();则不会受到任何的影响。

synchronized(this)

上面的例子中发现了锁的对象起的效果只是一个让线程之间竞争令牌,实际开发中如果随便找一个全局属性作为对象锁的话也不太合适,但如果为synchronized专门搞一个对象来锁更不划算,所以synchronized(this)是一个很好的选择。

public Object obj;

public void fun(){
    obj.funA();
    synchronized(this){
       obj.funB();
    }
}

该方法与上节代码等价。

锁一个类

值得注意的是,上面不管是synchronized(obj)还是synchronized(this)他们锁的作用范围都是当前对象的实例,不管有多少个线程,都是自己对象实例的锁自己的对象用,其他对象实例的事管不着。

对象锁

上图两个线程(T1,T2)中分别有两个对象实例(a1,a2),他们同时竞争synchronized(this)的作用范围是同一个对象实例。

但是还有一个锁,它的作用范围是整个类:synchronized(A.class)

使用synchronized(A.class)锁后,当前类下的所有对象实例都需要同时竞争使用权。

类锁

静态锁

在Java中静态属性和静态方法都是属于类的,而不是属于对象实例的,所以给静态属性和静态方法上锁都属于给类上锁,同上节类锁 一样。