synchronized真的很重么?

synchronized 是java中常见的保证多线程访问共享资源时的安全的一个关键字。很多人在讲到synchronized 时都说synchronized 是一把重量级的锁,那么synchronized 真的很重么?

synchronized 在jdk 1.6以前(不包括1.6)的确是一把很重的锁,每次使用锁的时候都是直接向操作系统请求的,所以效率低,且占资源,但是在jdk1.6以后,jvm对synchronized 进行了优化,加入了锁升级的功能,使得synchronized 在一些情况下并不是一把重量级的锁,而是一个很轻的一把锁。

本文就来探讨一下,jdk对synchronized 优化,包括锁升级、锁粗化、锁消除。

一、锁升级

锁升级其实是指,随着多线程并发加锁的程度提高而相应的对锁的状态的升级,可以分为:偏向锁、轻量级锁、自旋锁、重量级锁;

偏向锁

偏向锁不是一把锁,而是代表了当前synchronized 锁状态。偏向锁是一把很轻的锁,当只有一个来线程加锁的时候,此时synchronized 锁就会变成偏向锁,偏向锁代表这个锁偏向这个线程,就是说当这个线程再次来加锁的时候,不需要再向操作系统申请资源,而是很快就能获取到锁,减少了申请锁的开销。

为什么需要偏向锁。因为在大多数情况下,多线程竞争同一把锁的情况是很少的,那么在没有多线程并发竞争的情况下,其实没必要再去向系统申请重量级锁了,我就用目前这把偏向锁就够了,因为申请重量级会耗费比较大的资源。

轻量级锁

当随着更多线程来加锁的时候,偏向锁就会无法满足使用的条件了,因为偏向锁认为加锁的线程只有一个。

那么多线程加锁有可能会出现这种情况。当会有两个及以上的线程来加锁,但是没有出现同时来竞争锁的情况,也就是说虽然有多个线程来加锁,可能会出现A线程加完锁之后释放了锁,此时B来加锁,发现并没有线程持有锁,也就说没有线程跟B来竞争,也就相当于多线程来交替加锁的情况。

当出现这种情况的时候,偏向锁就会升级为轻量级锁。轻量级锁就是指虽然可能会出现多线程来加锁的情况,但是并不存在锁竞争的情况,并不会存在锁冲突。

自旋锁

上面说到随着加锁的线程变多,出现了多线程交替加锁的情况,偏向锁会升级为轻量级锁,但是随着并发加锁的线程越来越多,出现了多个线程同时来加锁的情况,也就不是交替加锁,那么此时轻量级锁已经不适合使用了,但是jvm为了防止锁直接升级为重量级锁(因为挂起线程和恢复线程的操作都需要转入到内核态中完成,这些操作给系统的并发性能带来很大的压力),加入了线程自旋的机制。所谓的自旋就是虽然加锁失败了,但是有可能出现其他线程很快就释放锁的情况,那么就尝试不断的自旋来尝试加锁,而不是直接将锁升级为重量级锁。

自旋锁的好处就是用来减少操作系统的压力。

重量级锁

但是随着加锁的线程不断增多,自旋了一定的时间或者次数也没有成功加锁,那么锁就会升级为重量级锁,因为自旋会消耗cpu,一直这么自旋也不是很好的选择,所以就会升级为重量级锁。升级为重量级锁之后,所有来加锁的线程加锁失败之后,就会加入等待的队列中,等待别的线程释放锁之后再进行锁的竞争。

通过以上的分析,我们也看出了,synchronized 在最开始的时候并不是上来就是一把重量级的锁,而是随着多线程并发竞争锁的激烈程度的提高来不断的升级,慢慢变成重量级锁,在整个升级的过程会经历偏向锁、轻量级锁、自旋锁、重量级锁的过程。

二、锁消除

先来看一段代

java;gutter:true; public class SynchronizedDemo {</p> <pre><code>public static void main(String[] args) { Object monitor = new Object(); synchronized (monitor){ System.out.println("加锁成功...."); } } </code></pre> <p>}</p> <pre><code> 通过上面代码我们可以看出,synchronized 的锁对象其实是方法的一个局部变量,那么对于这种情况,不会出现多线程竞争同一把Object 对象锁的情况,那么在运行的时候就会忽略这个synchronized 加锁的过程。 说的官方一点就是指虚拟机即时编译器在运行时,对一些代码上要求同步,但是被检测到不可能存在共享数据竞争的锁进行消除。 ## 三、锁粗化 ;gutter:true;
public class SynchronizedDemo {

private static final Object MONITOR = new Object();

public static void main(String[] args) {

for (int i = 0; i < 10; i++) {
synchronized (MONITOR) {
System.out.println("加锁成功….");
}
}
}
}

通过这段代码的分析,可以看出,循环内部每次都对同一个对象进行加锁和解锁,对于这种一串零碎的操作都对同一个对象加锁情况,虚拟机将会把加锁同步的范围扩展 (粗化)到整个操作序列的外部。以上述代码为例,也就是扩展到把for循环这个操作加锁,这样只需要加锁一次就可以了。

总结

所以,通过本篇文章可以看出,jdk对synchronized 其实进行了一系列的优化来尽可能减少加锁时对于性能的消耗,包括锁升级、锁消除、锁粗化。希望通过本篇文章可以让你对synchronized 底层又有一个全新的认识。

如果觉得这篇文章对你有所帮助,还请帮忙点赞、在看、转发一下,码字不易,非常感谢!

如果觉得这篇文章对你有所帮助,还请帮忙点赞、在看、转发给更多的人,码字不易,非常感谢!

往期热门文章推荐

扫码或者搜索关注公众号 三友的java日记 ,及时干货不错过,公众号致力于通过画图加上通俗易懂的语言讲解技术,让技术更加容易学习。

synchronized真的很重么?

Original: https://www.cnblogs.com/zzyang/p/16336060.html
Author: 三友的java日记
Title: synchronized真的很重么?

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/621752/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • Spring MVC之@RequestParam @RequestBody @RequestHeader 等详解

    (转自:http://blog.csdn.net/walkerjong/article/details/7946109#) 引言: 接上一篇文章,对@RequestMapping进…

    Java 2023年5月30日
    056
  • 【主流技术】ElasticSearch 在 Spring 项目中的实践

    前言 ElasticSearch简称es,是一个开源的高扩展的分布式全文检索引擎。 它可以近乎实时的存储、检索数据,其扩展性很好,ElasticSearch是企业级应用中较为常见的…

    Java 2023年6月6日
    088
  • SpringBoot:SpringBoot引入外部依赖包并打包的解决方式

    由于项目需求,需要引用外部依赖jar包,并打进生成的项目jar包内。 pom中加入如下配置: 通过Maven导入外部jar包: scope: 配置的参数作用是 system:被依赖…

    Java 2023年5月30日
    069
  • IaaS/ PaaS/ SaaS

    PaaS(平台即服务) 如 低代码平台IaaS (基础架构即服务) 如 阿里云主机SaaS (软件即服务) 如 淘宝之于卖家 Original: https://www.cnblo…

    Java 2023年6月15日
    068
  • ASP.NET Core GRPC 和 Dubbo 互通

    一.前言 Dubbo 是比较流行的服务治理框架,国内不少大厂都在使用。以前的 Dubbo 使用的是私有协议,采集用的 hessian 序列化,对于多语言生态来说是极度的不友好。现在…

    Java 2023年6月8日
    0136
  • jq命令用法总结

    原创:扣钉日记(微信公众号ID:codelogs),欢迎分享,转载请保留出处。 如果说要给Linux文本三剑客(grep、sed、awk)添加一员的话,我觉得应该是jq命令,因为j…

    Java 2023年6月7日
    073
  • git pull与git pull –rebase

    link:date: 2022-08-30 git pull –rebase 等效命令 总结 参考文章 git pull –rebase 在 push 代码…

    Java 2023年6月13日
    066
  • GBase8t客户端与服务器的通信

    通信方式支持类型: 共享内存(Shared memory) 流管道(Stream pipe) TCP/IP通信 IPX IPX/SPX DRDA 说明: gbase8t一个实例支持…

    Java 2023年6月9日
    034
  • 人到中年,做管理真的需要懂的管理必备知识

    课堂三点要求: 认真听讲,记笔记 * – 讲义电子版会给补充,不要急于找资料 – 跟着课堂节奏 积极参与课堂互动,远程依然有温度 * – 课堂提…

    Java 2023年6月16日
    068
  • MySQL之事务和redo日志

    事务 事务的四个ACID特性。 Atomicity 原子性 Consistency 一致性 Isolation 隔离性 Durability 持久性 原子性 原子性即这个事务的任务…

    Java 2023年6月16日
    090
  • SpringBoot添加过滤器、拦截器学习笔记

    SpringBoot添加过滤器、拦截器 添加过滤器有两种方式: * 1. 过滤器继承Filter并在类上面添加@WebFilter,启动类上面加上@ServletComponent…

    Java 2023年6月6日
    087
  • Spring boot Access-Control-Allow-Origin 问题解决

    import org.springframework.context.annotation.Bean; import org.springframework.context.ann…

    Java 2023年5月30日
    055
  • Java封装Get/Post类

    封装的类: package pers.hmi.translate; import java.io.BufferedReader; import java.io.IOExceptio…

    Java 2023年6月9日
    079
  • 机器学习基础知识

    ROC曲线含义 DL 卷积运算:参考吴恩达课程中的边缘检测,滑动窗口,乘法加法 运算步骤:滑动窗口,逐个相乘再相加 运算结果:新的矩阵,(s-f)/2 + 1 padding: 在…

    Java 2023年6月7日
    077
  • Linux命令三剑客

    1.awk AWK&#x662F;&#x4E00;&#x79CD;&#x5904;&#x7406;&#x6587;&#x67…

    Java 2023年6月5日
    080
  • 【Unity】在Unity中通过触发播放音频

    1、创建音频播放组件 首先在Hierarchy窗口中新建空组件。 选中新建的组件,在Inspector窗口中添加组件”Audio Source”,并将要播放…

    Java 2023年6月9日
    096
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球