synchronized的作用是同步一段代码
作用在方法上
public synchronized void method(){
}
作用于一段代码块
public void method1(){
synchronized(this) {
method();
}
}
synchronized的底层实现使用的是monitor(监视器)的东西,这个monitor是各个线程争夺的主角,获得monitor的线程拥有执行代码的权利。
java语言的顶级父类Object持有monitor,意味着java中所有的类,都是拥有monitor的。
我们可以像理解属性一样理解monitor,比方说非静态属性属于各自的对象,那么非静态的对象持有的monitor也只属于这个对象;同时,静态的属性属于一个类,那么静态类持有的monitor则也属于这个类。
monitor属于一个对象,称为对象锁;属于类,称为类锁。
方法上的锁:
有了上述理解,可以判断:
当synchronized作用在非静态方法上时,默认持有的锁属于this(当前对象),是对象锁。线程持有了对象锁,就可以执行所有被当前对象锁保护的代码。
public synchronized void method(){
}
当synchronized作用在静态方法上时,默认持有的锁属于class(静态类),是类锁。线程持有了类锁,就可以执行所有被类锁保护的代码。
public class Foo{
public synchronized static void method(){
}
}
代码块的锁:
当synchronized作用于代码块时,持有的是括号里的锁。
如果括号里是一个对象,就是对象锁。如果是class,就是类锁。
private Foo foo = new Foo();
public void method1(){
synchronized(foo) {
method();
}
synchronized(Foo.class) {
method();
}
}