版权声明:可以转载,请注明链接出处 https://blog.csdn.net/xihuanyuye/article/details/83959725
synchronized用来处理多个线程同时访问同一个类的一个代码块、方法,甚至这个类。
(1)修饰代码块时,需要设置一个参考对象作为锁的对象(对象锁)。
(2)修饰方法时,默认是当前对线作为锁的对象。
(3)修饰类时,默认是当前类的Class对象作为锁的对象。
1、对象锁
当一个对象中有同步方法或者同步块,线程调用此对象进入该同步区域时,必须获得对象锁。如果此对象的对象锁被其他调用者占用,则进入阻塞队列,等待此锁被释放(同步块正常返回或者抛异常终止,由JVM自动释放对象锁)。
注意,方法锁也是一种对象锁。当一个线程访问一个带synchronized方法时,由于对象锁的存在,所有加synchronized的方法都不能被访问(前提是在多个线程调用的是同一个对象实例中的方法)。
默认就是锁定对象的方法锁;
public class object {
public synchronized void method(){
System.out.println("我是对象锁也是方法锁");
}
}
需要设置具体对象的代码块锁
public class object {
public void method(){
synchronized(this){
System.out.println("我是对象锁");
}
}
}
2、类锁
一个class其中的静态方法和静态变量在内存中只会加载和初始化一份,所以,一旦一个静态的方法被申明为synchronized,此类的所有的实例化对象在调用该方法时,共用同一把锁,称之为类锁。
也是两种形式:
对应到对象锁,也是看锁定的位置是否需要显示指定对象名。类的对象就是Object;
public class object {
public static synchronized void method(){
System.out.println("我是第一种类锁");
}
}
需要显示指定锁定对象的代码块锁
public class object {
public void method(){
synchronized (object) {
System.out.println("我是第二种类锁");
}
}
}
3. synchronized位置的区别
在对象锁或者类锁中,synchronized的位置就是锁的大小,写在类名上的synchronized比写在代码块上明显颗粒度要大很多。效率也会更低。
参考:
方法锁、对象锁和类锁区别