首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往
  • 您找到你想要的搜索结果了吗?
    是的
    没有找到

    【JAVA 进阶之锁机制】synchronized 的锁升级-持续更新....

    1、synchronized 的基本认识 场景:Synchronized是一个同步关键字,在某些多线程场景下,如果不进行同步会导致数据不安全,而Synchronized关键字就是用于代码同步。什么情况下会数据不安全呢,要满足两个条件:一是数据共享(临界资源),二是多线程同时访问并改变该数据。 在多线程并发编程中 synchronized 称呼为重量级锁。但是,随着 Java SE 1.6 对 synchronized 进行了各种优化之后,有些情况下它就并不那么重,Java SE 1.6 中为了减少获得锁和释放锁带来的性能消耗而引入的偏向锁和轻量级锁。 1.1 synchronized 有三种方式来加锁 1.1.1. 修饰实例方法,作用于当前实例加锁,进入同步代码前 要获得当前实例的锁 1.1.2. 静态方法,作用于当前类对象加锁,进入同步代码前要 获得当前类对象的锁 1.1.3. 修饰代码块,指定加锁对象,对给定对象加锁,进入同 步代码库前要获得给定对象的锁。 1.2 Mark word Mark word 记录了对象和锁有关的信息,当某个对象被 synchronized 关键字当成同步锁时,那么围绕这个锁的一 系列操作都和 Mark word 有关系。Mark Word 在 32 位虚 拟机的长度是 32bit、在 64 位虚拟机的长度是 64bit。 Mark Word 里面存储的数据会随着锁标志位的变化而变化, Mark Word 可能变化为存储以下 5 中情况

    03

    Java并发机制(-synchronized的实现原理与应用)

    1 synchronized的实现原理与应用 1.1 偏向锁和轻量级锁以及锁的存储结构和升级过程 1.1.1 实现同步的基础,Java中每一个对象都可以作为锁 a:对于普通同步方法,锁是当前实例对象。 b:对于静态同步 方法,锁是当前类的Class对象。 c:对于同步方法块,锁是synhronized 括号里配置的对象。 1..2 Java对象头: 1..2.1 synchronized 用的锁是存在Java对象头李的。 1.3 锁的升级与对比 为了减少获得锁和释放锁带来的性能消耗,引入了偏向所和轻量级锁。在JavaSE 1.6 中,级别从低到高依次是:无锁状态,偏向锁状态,轻量级锁状态和重量级锁状态。这几个状态会随着竞争情况逐渐升级。锁可以升级丹不能降级。目的是为了提高获得锁和释放锁的效率。 2.1 偏向锁。使用机制:等到竞争出现才释放锁的机制。偏向所在java6和Java7里是默认启用的,如果你确定应用程序里所有的锁通常情况下处于竞争状态,可以通过JVM参数关闭偏向锁,-XX:-UseBiasedLocking=false,那么默认会进入轻量级锁状态。 2.2 轻量级锁 2.2.1 轻量级锁加锁:当前线程便尝试使用自旋来获取锁。 2.2.2 轻量级锁解锁:若果锁存在竞争,锁就会膨胀成重量级锁。因为自旋会消耗cpu,为了避免无用的自旋,比如获得锁的线程被阻塞住了,一旦锁升级为重量级锁,就不会回复到轻量级锁i,当锁处于这个状态下,卡线程视图获得锁时,都会被阻塞住。锁的优缺点对比: 偏向锁:加锁和解锁不需要额外的消耗,和执行非同步方法相比仅存在纳秒级的差距。-优点 如果线程间存在锁竞争,会带来额外的锁撤销的消耗-缺点 适用于只有一个线程访问同步快场景。-使用场 轻量级锁:竞争的线程不会阻塞,提高了程序的响应速度。 如果始终得不到锁竞争的线程,使用自旋会消耗cpu 追求响应时间 同步块执行速度非常快 重量级锁:线程竞争不适用自旋,不会消耗cpu 线程阻塞,响应时间缓慢。 追求吞吐量 同步块执行速度较长

    02

    Synchronized 偏向锁、轻量级锁、自旋锁、锁消除

    上篇文章中向大家介绍了Synchronized的用法及其实现的原理。现在我们应该知道,Synchronized是通过对象内部的一个叫做监视器锁(monitor)来实现的。但是监视器锁本质又是依赖于底层的操作系统的Mutex Lock来实现的。而操作系统实现线程之间的切换这就需要从用户态转换到核心态,这个成本非常高,状态之间的转换需要相对比较长的时间,这就是为什么Synchronized效率低的原因。因此,这种依赖于操作系统Mutex Lock所实现的锁我们称之为“重量级锁”。JDK中对Synchronized做的种种优化,其核心都是为了减少这种重量级锁的使用。JDK1.6以后,为了减少获得锁和释放锁所带来的性能消耗,提高性能,引入了“轻量级锁”和“偏向锁”。

    01

    Java并发编程:Synchronized底层优化(偏向锁、轻量级锁)

    上篇文章中向大家介绍了Synchronized的用法及其实现的原理。现在我们应该知道,Synchronized是通过对象内部的一个叫做监视器锁(monitor)来实现的。但是监视器锁本质又是依赖于底层的操作系统的Mutex Lock来实现的。而操作系统实现线程之间的切换这就需要从用户态转换到核心态,这个成本非常高,状态之间的转换需要相对比较长的时间,这就是为什么Synchronized效率低的原因。因此,这种依赖于操作系统Mutex Lock所实现的锁我们称之为“重量级锁”。JDK中对Synchronized做的种种优化,其核心都是为了减少这种重量级锁的使用。JDK1.6以后,为了减少获得锁和释放锁所带来的性能消耗,提高性能,引入了“轻量级锁”和“偏向锁”。

    02

    Java Concurrent 偏向锁&轻量级锁&重量级锁

    再说偏向锁之前先来看一下Java 对象头,Java 对象是分为 对象头、实例数据、对齐填充三部分,创建一个Java 对象所消耗和占用的cpu和内存代价都是很高的(尤其是对齐填充这一块,真的会浪费很多内存),和并发相关性最大的是对象头,因为Java 原生锁(sychronized)的信息是存放在Java 对象头中的。如果对象是数组类型,则虚拟机用3个Word(字宽)存储对象头,如果对象是非数组类型,则用2字宽存储对象头。 对象头中的位数依赖于系统的位数: 1、32或64bit存放Mark Word,其中包括存储对象的hashCode或锁信息等。 2、32或64bit存放Class Metadata Address,也就是存储到对象类型数据的指针。 3、如果是数组对象的话,使用32或64bit存放Array length,也就是数组的长度)

    02
    领券