从轻量级锁 来看锁机制。
(目前 上的唯一一张图。= =。 因为有些东西没有图的话 是很难理清楚的 - - )
比如当对象头未锁定时候 存储的是对象哈希码和分代年龄。锁标志位为01 。而锁被占用时候存储的指向锁记录的指针以及锁标志位。
synchronized关键字在jdk版本更迭过程中不断被优化,对象的加锁过程不再是直接加最原始的重量级锁,而是先从偏向锁开始, 会升级为轻量级锁,最终变成重量级锁。 这里只总结一下轻量级锁的。
轻量级锁的加锁:
1. 当进入同步代码块的时候 如果该对象未被锁定,即标志位是01 。那么就会为当前线程的栈帧中建立一个 lock record 锁记录空间,其实就是markword的拷贝
2. 然后 执行cas操作,也就是将markword与栈帧中锁记录比对 如果比对成功则立刻把markword更新为指向锁记录的指针,这整个过程都是cas操作。 就表示该线程拥有了对象锁。将标志位转变为00
这里,假设多个线程同时这么做,由于是原子操作,那么肯定仅有一个线程能实现更新markword的值为指向锁记录指针的操作。其他线程cas 肯定就会失败, 因为 markword的值已经变了。
3. 如果更新操作失败了,会检查对象markword 是否指向当前线程的栈帧,如果是的话说明已经拥有了该锁,如果不是的话,则是其他线程 持有的该对象锁。
这个流程有个问题