面试官:谈一谈你对synchronized的理解

Java中对象和Monitor

对象

对象中有对象头,实例数据

对象头(mark word)长啥样

hashcode,分代年龄,是否是偏向锁,加锁标识

Monitor

monitor 又称管程,也叫锁。是JVM层提供的,每一个对象对应一个monitor。

工作原理

  • 用了synchronized(obj)之后,对象的markword中就有个指针指向monitor的地址,并且monitor中的ownernull变为thread-1.
  • thread-2来抢占共享资源,发现monitor 已经被抢占了,就去阻塞队列里面排队,变成阻塞状态.
  • synchronized(obj) 执行完之后会唤醒阻塞队列中的线程,此时是非公平

轻量级锁

  • 加锁 每个线程在加锁的时候会产生一个锁记录,锁记录中的 lock-recoad-address 00 与 对象头中的markword cas交换,轻量级锁加锁成功
  • 冲突 同一个线程重新获取锁,会产生一个新的锁记录,因为已经加锁了,就cas不成功,锁记录的地址为null
  • 解锁 锁记录为null的话就表示是重入的锁,不为null就交换markword.

锁膨胀

  • 加锁 在刚才的冲突环节,如果不是自己的线程,那么会锁就会膨胀,对象的markword就指向Monitor地址 10 此时thread-1就进入entryList 阻塞队列中,等待唤醒.
  • 解锁 等thread-0 执行完之后,发现cas交换失败了,就会进行重量级锁的解锁,将Monitor 中的owner赋值null,并且唤醒EntryList中的线程.

偏向锁

  • 加锁 我们发现轻量级锁每次同一个线程还是需要cas检查,这也是很耗费性能的,JVM对这个进行了优化,默认是开启偏向锁.不采用锁记录和对象头交换的形式,而是直接将线程的地址和对象的markword交换,这样每次只需要看当前来的线程是本线程就不需要cas了.
相关推荐
快乐就去敲代码@!27 分钟前
Boot Cache Star ⭐(高性能两级缓存系统)
spring boot·redis·后端·缓存·docker·压力测试
我命由我1234539 分钟前
Python Flask 开发 - Flask 路径参数类型(string、int、float、path、uuid)
服务器·开发语言·后端·python·flask·学习方法·python3.11
文心快码BaiduComate39 分钟前
Comate强力赋能:「趣绘像素岛」从体验泥潭到高性能可用的蜕变之路
前端·后端·程序员
阿杆.1 小时前
如何在 Spring Boot 中接入 Amazon ElastiCache
java·spring boot·后端
Apifox1 小时前
Apifox + AI:接口自动化测试的智能化实践
前端·后端·测试
别惹CC1 小时前
Spring AI 进阶之路04:集成 SearXNG 实现联网搜索
java·后端·spring
invicinble1 小时前
springboot的日志体系
java·spring boot·后端
PFinal社区_南丞1 小时前
现代PHP开发实战
后端·php
徐小夕1 小时前
10k Star 的开源 AI 记忆引擎:6 行代码,用图谱+向量打造永不遗忘的 AI
前端·后端·github
czlczl200209251 小时前
拒绝 DTO 爆炸:详解 Spring Boot 参数校验中的“分组校验” (Validation Groups) 技巧
java·spring boot·后端