什么是synchronized关键字
咱们先来瞅瞅官方的原话:
Synchronized methods enable a simple strategy for preventing thread interference and memory consistency errors: if an object is visible to more than one thread, all reads or writes to that object’s variables are done through synchronized methods.
地址: https://docs.oracle.com/javase/tutorial/essential/concurrency/syncmeth.html
让我粗略的翻译一下上文的意思,大概如下:
Synchronized 方法提供了一种简单策略来防止线程干扰和内存一致性错误:如果一个对象对多个线程是可见的,那么对该对象变量的所有读取或写入均通过同步方式完成。
白话文:
synchronized提供了一种排他机制,也就是在同一时间只能有一个线程去执行某些操作
具体表现如下:
- synchronized关键字提供了一种锁的机制,能够确保共享变量的互斥访问,从而防止数据不一致问题的出现
- synchronized关键字包括monitor enter 和 monitor exit两个JVM指令,它能够保证在任何时候任何线程执行到monitor enter成功之前都必须从主内存中获取数据,而不是从缓存中,在monitor exit运行成功之后,共享变量被更新后的值必须刷入主内存
- synchronized的指令严格遵守java happens-before规则 ,一个monitor exit指令之前 必定要有一个 monitor enter
public class SynchronizedCounter {
private int c = 0;
public synchronized void increment() {
c++;
}
public synchronized void decrement() {
c--;
}
public synchronized int value() {
return c;
}
}
则使这些方法同步具有两个效果:
首先,不可能对同一对象的两次同步方法调用进行交织。当一个线程正在执行对象的同步方法时,所有其他调用同一对象块的同步方法的线程(挂起执行),直到第一个线程对该对象完成。
其次,当同步方法退出时,它将自动与之前对同一对象的同步方法的任何调用建立先发生关系。这保证了对象状态的更改对所有线程都是可见的。
请注意,构造函数无法同步-将synchronized关键字与构造函数一起使用是语法错误。同步构造函数没有任何意义,因为只有创建对象的线程才可以在构造对象时对其进行访问。
普通方法锁住当前对象,静态方法锁住Class对象,代码块,锁定()中对象;通过jvm 对象监视器Monitor实现