synchronized关键字

锁一个对象

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

public Object obj;

public void fun(){
    obj.funA();
    synchronized(obj){
       obj.funB();
    }
}
复制代码

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

synchronized(this)

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

public Object obj;

public void fun(){
    obj.funA();
    synchronized(this){
       obj.funB();
    }
}
复制代码

该方法与上节代码等价。this

锁一个类

值得注意的是,上面无论是synchronized(obj)仍是synchronized(this)他们锁的做用范围都是当前对象的实例,无论有多少个线程,都是本身对象实例的锁本身的对象用,其余对象实例的事管不着。spa

对象锁

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

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

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

类锁

静态锁

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