在Java中,synchronized 关键字确实用于实现同步,但同步的范围是在进入和离开同步块时,而不是在获取锁的瞬间。具体来说:
获取锁:
偏向锁: 当一个线程第一次进入同步块时,JVM会将锁偏向于这个线程,将线程ID保存在对象头中。后续进入同步块的时候,不需 要再进行CAS操作,直接判断线程ID即可。
轻量级锁: 当有多个线程竞争同一个锁时,会升级为轻量级锁,JVM使用CAS操作尝试获取锁。如果成功,线程进入临界区,如果失败(通常是10次仍然无法获取锁时),就升级为重量级锁。升级重量级锁以后 就会阻塞线程
注:CAS的原理是(Compare And Swap)操作中,如果两个线程同时尝试修改同一个共享数据,只有一个线程的CAS操作会成功,而另一个线程的CAS操作将失败。这是因为CAS的成功与否依赖于当前内存位置的值是否与期望值相等)
CAS操作包含三个基本操作:比较、更新和返回结果。具体步骤如下:
读取内存中的旧值。
比较内存中的旧值与预期值。
如果比较成功,则将内存中的值更新为新值,否则不执行任何操作。
返回操作结果,指示操作是否成功。
执行同步块:
一旦线程成功获取锁,它就可以执行被 synchronized 保护的方法或代码块。在同步块内部,线程是独占锁的,其他线程如果想要进入同步块,必须等待当前线程执行完毕并释放锁。
释放锁:
当线程执行完 synchronized 保护的方法或代码块时,它会释放锁,这样其他线程就有机会获取锁并执行相应的同步块。
关键点在于同步块内的代码是由获得锁的线程单独执行的,其他线程需要等待锁的释放才能进入同步块。这确保了在同一时刻,只有一个线程能够执行被 synchronized 保护的关键代码,从而实现了对共享资源的互斥访问,确保线程安全性。
总的来说,synchronized 关键字确保了获取锁和执行同步块之间的原子性,从而实现了同步。