随便写写,方便记忆,可能有误,谢指正。
一、synchronized使用方式
1.实例方法同步:
同步是锁在拥有该方法的对象上。对象锁是每个对象各有一把的。所以如果要实现同步,则需要请求都使用同一个实例,然后进行加锁。
public synchronized void xxx(int x){
this.y += x;
}
2.静态方法同步:
静态方法的同步是锁在该方法所在的类对象上。是用来锁类的,一个类的所有对象共享一个class对象,共享同一个静态方法,类锁的作用就是使持有者可以同步地调用静态方法。当synchronized指定修饰静态方法或者class对象的时候,拿到的就是类锁,类锁是所有对象共同争抢一把。
public static synchronized void xxx(int x){
y += value;
}
3.实例方法中的同步块:
同步实例方法使用调用方法本身的实例作为监视器对象(锁对象),对象锁是每个对象各有一把的。所以如果要实现同步,则this对象必须是同一个给类的实例,才能实现同步。
public void xxx(int x){
synchronized(this){
this.y += x;
}
}
4.静态方法中的同步块:
方法同步在该方法所属的类对象上。是用来锁类的,一个类的所有对象共享一个class对象,共享同一个静态方法,类锁的作用就是使持有者可以同步地调用静态方法。当synchronized指定修饰静态方法或者class对象的时候,拿到的就是类锁,类锁是所有对象共同争抢一把。
public static void xxx(String x, String z){
synchronized(XXX.class){
y = x + z;
}
}
类锁和对象锁同时存在,两者不能做到同步,因为是不同的锁。
二、synchronized用在方法上和代码块中的区别
Java虚拟机中的同步(Synchronization)基于进入和退出Monitor对象实现,无论是显式同步(有明确的 monitorenter 和 monitorexit 指令,即同步代码块)还是隐式同步都是如此。
在Java语言中,同步用的最多的地方可能是被 synchronized 修饰的同步方法。同步方法并不是由 monitorenter 和 monitorexit 指令来实现同步的,而是由方法调用指令读取运行时常量池中方法表结构的 ACC_SYNCHRONIZED 标志来隐式实现的。
同步代码块:monitorenter 指令插入到同步代码块的开始位置,monitorexit 指令插入到同步代码块的结束位置,JVM需要保证每一个 monitorenter 都有一个 monitorexit 与之相对应。
任何对象都有一个 monitor 与之相关联,当且一个 monitor 被持有之后,他将处于锁定状态。线程执行到 monitorenter 指令时,将会尝试获取对象所对应的 monitor 所有权,即尝试获取对象的锁;