Java中Synchronized和Lock的底层实现到底有什么区别

【1】synchronized

synchronized通过monitor和mutex lock实现了互斥

synchronized是通过每个对象与之相关的一个叫做监视器(monitor)来实现的
监视器内有个组件是计数器,默认值为0,

  • 当计数器为0,尝试加锁,加锁成功,计数器加1(本线程可以多次获取这个锁,可重入
  • 其他线程发现计数器不为0,加锁失败进入阻塞

监视器锁本质是依赖于底层的操作系统的Mutex Lock是互斥锁来实现的
Mutex Lock指令的调用需要从用户态转换到核心态
成本非常高,造成synchronized效率低
因此synchronized称其为“重量级锁”

JDK对synchronized进行了优化,引入了膨胀的过程: 无锁->偏向锁->轻量级锁->重量级锁即Mutex Lock


【2】Lock

Lock的唯一实现类:ReetrantLock,依赖AQS来实现
Java 中Lock接口提供了ReetrantLock实现类来实现可重入锁功能。ReetrantLock基于AQS (抽象队列同步器)来实现加锁的获取、可重入、释放。实现可重入的功能: 可以重复地获取自己(本线程)所拥有的锁。

ReentrantLock 的底层原理是怎样的?

锁内部有两个核心参数:

  1. state:计数器
  2. exclusiveOwnerThread:所有者线程

第一步:先进行CAS算法,CAS是原子性的操作同时只能一个线程,当state计数值为0时,这个锁就被认为是没有被任何线程所占有的,其他线程会进入等待队列

第二步:当线程请求一个未被持有的锁时,计数值将会递增

第三步:而当线程退出同步代码时,计数器会相应地递减。当计数值为0时,则释放该锁


【3】synchronized 和 Lock 有什么区别?

  1. 实现层面不一样。synchronized 是 Java 关键字,JVM层面 实现加锁和释放锁(代码中没有,但进行反编译会发现monitorentermonitorexit);Lock 是一个接口,ReentrantLock在代码层面实现加锁和释放锁
  2. 是否自动释放锁。synchronized 在线程代码执行完或出现异常时自动释放锁;Lock 不会自动释放锁,需要再 finally {} 代码块显式地中释放锁
  3. 是否一直等待。synchronized 会导致线程拿不到锁一直等待;Lock 可以设置尝试获取锁或者获取锁失败一定时间超时
  4. 获取锁成功是否可知。synchronized 无法得知是否获取锁成功;Lock 可以通过tryLock 获得加锁是否成功
  5. 功能复杂性。synchronized 加锁可重入、不可中断、非公平;Lock 可重入、可判断、可公平和不公平、细分读写锁提高效率

类别synchronizedLock
存在的层次Java关键字 在JVM层面上是一个类
锁的释放1以获取锁的线程执行完同步代码,释放锁 2线程执行发生产异常,jvm会让线程释放锁在finally中必须释放锁,不然容易造存线程死锁发生异常
锁的获取假设A线程获取锁,B线程等待。如果A线程阻塞,B线程会一直等待可以尝试获得锁,线程可以不用一直等待
锁状态无法判断可以判断
锁类型可重入、不可中断、非公平可重入、可判断 可公平 、可不公平
锁的性能少量同步大量同步
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
面试题包括以下十九部分:Java 基础、容器、多线程、反射、对象拷贝、Java Web 模块、异常、网络、设计模式、Spring/Spring MVC、Spring Boot/Spring Cloud、Hibernate、Mybatis、RabbitMQ、Kafka、Zookeeper、MySql、Redis、JVM 。 目录: 一、Java 基础 1.JDK 和 JRE 有什么区别? 2.== 和 equals 的区别是什么? 3.两个对象的 hashCode()相同,则 equals()也一定为 true,对吗? 4.final 在 java 有什么作用? 5.java 的 Math.round(-1.5) 等于多少? 6.String 属于基础的数据类型吗? 7.java 操作字符串都有哪些类?它们之间有什么区别? 8.String str="i"与 String str=new String(“i”)一样吗? 9.如何将字符串反转? 10.String 类的常用方法都有那些? 11.抽象类必须要有抽象方法吗? 12.普通类和抽象类有哪些区别? 13.抽象类能使用 final 修饰吗? 14.接口和抽象类有什么区别? 15.java IO 流分为几种? 16.BIO、NIO、AIO 有什么区别? 17.Files的常用方法都有哪些? 二、容器 18.java 容器都有哪些? 19.Collection 和 Collections 有什么区别? 20.List、Set、Map 之间的区别是什么? 21.HashMap 和 Hashtable 有什么区别? 22.如何决定使用 HashMap 还是 TreeMap? 23.说一下 HashMap 的实现原理? 24.说一下 HashSet 的实现原理? 25.ArrayList 和 LinkedList 的区别是什么? 26.如何实现数组和 List 之间的转换? 27.ArrayList 和 Vector 的区别是什么? 28.Array 和 ArrayList 有何区别? 29.在 Queue poll()和 remove()有什么区别? 30.哪些集合类是线程安全的? 31.迭代器 Iterator 是什么? 32.Iterator 怎么使用?有什么特点? 33.Iterator 和 ListIterator 有什么区别? 34.怎么确保一个集合不能被修改? 三、多线程 35.并行和并发有什么区别? 36.线程和进程的区别? 37.守护线程是什么? 38.创建线程有哪几种方式? 39.说一下 runnable 和 callable 有什么区别? 40.线程有哪些状态? 41.sleep() 和 wait() 有什么区别? 42.notify()和 notifyAll()有什么区别? 43.线程的 run()和 start()有什么区别? 44.创建线程池有哪几种方式? 45.线程池都有哪些状态? 46.线程池 submit()和 execute()方法有什么区别? 47.在 java 程序怎么保证多线程的运行安全? 48.多线程的升级原理是什么? 49.什么是死? 50.怎么防止死? 51.ThreadLocal 是什么?有哪些使用场景? 52.说一下 synchronized 底层实现原理? 53.synchronized 和 volatile 的区别是什么? 54.synchronizedLock 有什么区别? 55.synchronized 和 ReentrantLock 区别是什么? 56.说一下 atomic 的原理? 四、反射 57.什么是反射? 58.什么是 java 序列化?什么情况下需要序列化? 59.动态代理是什么?有哪些应用? 60.怎么实现动态代理? 五、对象拷贝 61.为什么要使用克隆? 62.如何实现对象克隆? 63.深拷贝和浅拷贝区别是什么? 六、Java Web 64.jsp 和 servlet 有什么区别? 65.jsp 有哪些内置对象?作用分别是什么? 66.说一下 jsp 的 4 种作用域? 67.session 和 cookie 有什么区别? 68.说一下 session 的工作原理? 69.如果客户端禁止 cookie 能实现 session 还能用吗? 70.spring mvc 和 struts 的区别是什么? 71.如何避免 sql 注入? 72.什么是 XSS 攻击,如何避免? 73.什么是 CSRF 攻击,如何避免? 七、异常 74.throw 和 throws 的区别? 75.final、finally、finalize 有什么区别? 76.try-catch-finally 哪个部分

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值