synchronized()_Synchronized关键字引出的多种锁
前言
Java 中的 synchronized关键字可以在多线程环境下用来作为线程安全的同步锁。本文不讨论 synchronized 的具体使用,而是研究下synchronized底层的锁机制,以及这些锁分别的优缺点。
一、synchronized机制
synchronized关键字是JAVA中常用的同步功能,提供了简单易用的锁功能。
synchronized有三种用法,分别为:
- 用在普通方法上,能够锁住当前对象。
- 用在静态方法上,能够锁住类
- 用在代码块上,锁住的是synchronized()里的对象
在JDK6之前,synchronized使用的是重量级锁制,在之后synchronized加入了锁膨胀机制,显著提升了synchronized关键字的效率。
基于synchronized关键字,我们来了解下几种类别的锁,并且讲解synchronized的锁膨胀机制。
synchronized锁是非公平锁。并且一个被synchronized锁住的对象或类,就是一把锁。
另外一提,所有锁都是存储在Java对象头里的,Java对象头里的Mark Word里默认存储对象的HashCode,分代年龄和锁标记位。也就是说Mark Word记录了锁的状态
二、锁膨胀机制与几类锁
锁膨胀是不可逆的
2.1 偏向锁
synchronized在JDK1.6以后默认开启偏向锁,synchronized最初都是偏向锁
表现:一个线程获取锁成功后,会在对象头里记录线程ID,以后该线程获取和释放锁都没有任何花费。(因为该锁已经被绑定在该线程上了,且在膨胀前不会改变),如果其他线程尝试获取这个锁,偏向锁将会膨胀为轻量锁。
优点:在只有一个线程使用锁的时候获取和退出锁没有任何花费
缺点:锁竞争激烈会很快升级为轻量锁,那么维持偏向锁的过程就是在浪费计算机资源。(不过因为偏向锁本身就很轻量,因此浪费的资源并不多)
小结:只有一个线程使用锁的情况下,synchronized使用的锁为偏向锁。
如果锁竞争激烈,可以通过配置JDK禁用偏向锁。
2.2 轻量锁
一把锁不止一个线程使用,则偏向锁膨胀为轻量锁
表现:线程获取轻量锁时,会直接用CAS修改对象头里锁的记录,如果修改失败,代表此时锁存在多个线程的竞争,轻量锁将会膨胀为重量锁。
优点:在线程之间使用锁不存在竞争时,一次CAS操作就能获取和退出锁
缺点:与偏向锁类似
小结:只要一把锁不止一个线程获取过,偏向锁就会膨胀为轻量锁。
2.3 重量锁
一把锁存在多线程竞争,则轻量锁开始自旋,自旋一定次数后仍没获取锁,则膨胀为重量锁(存在竞争时,轻量锁虽然会先自旋,但是最终往往都会膨胀为重量锁)
表现:线程获取重量锁时,如果获取失败(即锁已被其他线程获取),则使用自适应自旋锁,自旋一定次数后仍没获取锁,则进入阻塞队列等待。
优点:未获取到的锁进入阻塞队列,节约CPU资源。(好吧感觉其实是没有啥优点)
缺点:重量锁是通过对象内部的监视器(monitor)实现,其中monitor的本质是依赖于底层操作系统的Mutex Lock实现,操作系统实现线程之间的切换需要从用户态到内核态的切换,切换成本非常高。
小结:只要一把锁存在多线程竞争,轻量锁就会膨胀为重量锁。
自旋锁
synchronized的轻量锁,重量锁,使用了自适应自旋锁进行性能优化
首先介绍自旋锁
表现:线程获取锁失败后,不会进入阻塞等待,而是再次尝试去获取锁,如此反复,直到获取到锁,或者自旋结束那么会阻塞等待。
解决问题:在某些场景下,线程持有锁的时间非常短。在线程获取锁失败后,如果线程进入阻塞将会带来线程上下文的切换,上下文切换的时间可能反而高于线程反复尝试获取锁的时间。
此时线程原地等待去重复获取锁。反而在性能上更有优势。
缺点:
- 单核CPU没有线程并行,反复尝试会导致进程无法继续运行。
- 重复尝试导致了CPU的占用,如果CPU资源紧张的话反而会性能下降
- 如果锁的竞争时间过长,不仅没有性能提升,还浪费了大量CPU资源。
优化:使用自适应自旋锁。自适应自旋锁会根据之前的锁获取记录,优化调整自旋时间,避免造成不必要的自旋。
三、具体synchronized流程
关注我,后续更多干货奉上
synchronized()_Synchronized关键字引出的多种锁相关推荐
- Java面试补给站——final、volatile、synchronized三大关键字
如果说语法是一个编程语言的骨架,那么关键字就是骨架的关节. 在Java中有各种功能的关键字,最常用的int,break,public这些关键字都是在编程过程中必不可少的. 本文从面试提问的角度聊一下f ...
- 使用synchronized和Lock对象获取对象锁
Java:使用synchronized和Lock对象获取对象锁 下面使用ReentrantLock这个锁来实现加锁功能: package com.tch.test.concurrent;import ...
- Java:使用synchronized和Lock对象获取对象锁
在并发环境下,解决共享资源冲突问题时,可以考虑使用锁机制. 1.对象的锁 所有对象都自动含有单一的锁. JVM负责跟踪对象被加锁的次数.如果一个对象被解锁,其计数变为0.在任务(线程)第一次给对象加锁 ...
- 为 什 么 说 Synchronized 是 一 个 悲 观 锁 ?
为 什 么 说 Synchronized 是 一 个 悲 观 锁 ?乐 观 锁 的 实 现 原 理又 是 什 么 ? 什 么 是 CAS, 它 有 什 么 特 性 ? a.Synchronized 显 ...
- linnux同步方法多种锁介绍
https://blog.csdn.net/zsf8701/article/details/7844316 1.互斥锁 2.自旋锁 3.读写锁 4.文件锁 5.条件变量 6.信号量 https://b ...
- 死磕synchronized五:系统剖析轻量级锁
哈喽,大家好,我是江湖人送外号[道格牙]的子牙老师. 近期准备写一个专栏:从Hotspot源码角度剖析synchronized.前前后后大概有10篇,会全网发,写完后整理成电子书放公众号供大家下载.对 ...
- synchronized原理_synchronized关键字的作用、原理以及锁优化
synchronized简介 synchronized块是Java提供的一种原子性内置锁,Java中的每个对象都可以隐式地扮演一个用于同步的锁的角色.这些Java内置的使用者看不到的锁被称为内部锁(i ...
- synchronized原理_synchronized 关键字底层原理
synchronized 关键字底层原理属于 JVM 层面. ① synchronized 修饰同步语句块 public class SynchronizedDemo {public void met ...
- 【并发编程】线程锁--Synchronized、ReentrantLock(可重入锁)
在说锁之前,我们要明白为什么要加锁,不加锁会怎样? 在并发编程中,很容易出现线程安全问题,接下来我们看个很经典的例子--银行取钱,来看一下有关线程安全的问题. 取钱的流程可以分为一下几个步骤: 1.用 ...
最新文章
- Cookie 位置_无需整理
- 微酒店微信界面返回信息有html源码
- [洛谷3811]【模板】乘法逆元
- java按钮位置_java窗口按钮位置设置
- 【类库】私房干货.Net数据层方法的封装
- Android Studio 安装ASM插件
- Ubuntu root账号的使用
- 计算机组成原理闭卷,《计算机组成原理》试卷A (闭卷)
- 阿里云建成全国最大数据中心集群 全面应用自研硬核技术
- CSDN创始人蒋涛出席2021 中关村论坛,启动“科创中国”开源创新榜单评选
- BZOJ3244 [Noi2013]树的计数 【数学期望 + 树遍历】
- 彻底解决四种路由器不能上网故障
- 11月19日 数据库连接 PDO
- python循环的基本思想是重复_python基础-循环
- alexa_Alexa可能进入Windows 10的锁定屏幕
- cad如何多选对象_cad中选择对象,不小心多选了一条线,怎么取消这个多选的家伙...
- 斐讯n1 无线打印服务器,斐讯N1 设置旁路由(网关)的方法 | nas2x
- 计算机知识怎么记忆,计算器记忆加怎么用
- jvm配置垃圾收集参数
- android adb 分析,android adb shell常用脚本分析课件.ppt
热门文章
- 升级drupal4.0.0_Drupal从1.0到8.0的简要历史
- go 初学者 错误_初学者的颠簸之旅,发现了一些不错的错误
- devops 开发_开发五年,DevOps手册问世
- input 底线_社区建设如何帮助组织的底线
- JavaScript canvas实现俄罗斯方块
- 第三十三章 机械化印刷
- Bootstrap表单控件的尺寸
- 让老IE支持CSS3响应式布局
- verp中joint的type、模式的区别
- php 7.2.5_CentOS7下PHP5.5.38升级PHP7.2.2