前言
synchronized 用来处理多个线程同时访问同一个类的一个代码块、方法,甚至这个类。
(1)修饰代码块时,需要一个reference对象作为锁的对象。
(2)修饰方法时,默认是当前对线作为锁的对象。
(3)修饰类时,默认是当前类的Class对象作为锁的对象。
1、方法锁
通过在方法声明中加入 synchronized关键字来声明 synchronized 方法。
synchronized 方法控制对类成员变量的访问:
每个类实例对应一把锁,每个 synchronized 方法都必须获得调用该方法的类实例的锁方能执行,否则所属线程阻塞,方法一旦执行,就独占该锁,直到从该方法返回时才将锁释放,此后被阻塞的线程方能获得该锁,重新进入可执行状态。这种机制确保了同一时刻对于每一个类实例,其所有声明为 synchronized 的成员函数中至多只有一个处于可执行状态,从而有效避免了类成员变量的访问冲突。
2、对象锁
当一个对象中有同步方法或者同步块,线程调用此对象进入该同步区域时,必须获得对象锁。如果此对象的对象锁被其他调用者占用,则进入阻塞队列,等待此锁被释放(同步块正常返回或者抛异常终止,由JVM自动释放对象锁)。
注意,方法锁也是一种对象锁。当一个线程访问一个带 synchronized 方法时,由于对象锁的存在,所有加 synchronized 的方法都不能被访问(前提是在多个线程调用的是同一个对象实例中的方法)。
对象锁有两种形式:
public class object {
public synchronized void method(){
System.out.println("我是对象锁也是方法锁");
}
}
另外一种是:
public class object {
public void method(){
synchronized(this){
System.out.println("我是对象锁");
}
}
}
3、类锁
一个class其中的静态方法和静态变量在内存中只会加载和初始化一份,所以,一旦一个静态的方法被申明为synchronized,此类的所有的实例化对象在调用该方法时,共用同一把锁,称之为类锁。
也是两种形式:
public class object {
public static synchronized void method(){
System.out.println("我是第一种类锁");
}
}
第二种:
public class object {
public void method(){
synchronized (object.this) {
System.out.println("我是第二种类锁");
}
}
}