Synchronized可重入锁分析
可重入锁又称递归锁,是指在同一个线程在外层方法获取锁的时候,再进入该线程的内层方法会自动获取锁(前提是锁对象必须是同一对象或者class),
不会因为之前已经获取过还没实方而发生阻塞。即同一线程可执行多个持有同一个锁的方法。
先来一段代码:
public class ReentrantSynchronized { public synchronized void firstIPhoneByElectric() { System.out.println("第一部iphone充电..." + Thread.currentThread().getName()); secondIPhoneByElectric(); } public synchronized void secondIPhoneByElectric() { System.out.println("第二部iphone充电..." + Thread.currentThread().getName()); } public static void main(String[] args) { ReentrantSynchronized reentrantSynchronized = new ReentrantSynchronized(); reentrantSynchronized.firstIPhoneByElectric(); } }
就像一个多用充电线可以同时给多部iphone充电,但是不会发生任何冲突和阻塞,也就是第一部iphone可以获取充电线(锁)充电的同时还可以第二部iphone也可以获取充电线(锁)进行充电。
执行结果如下:
在上面的代码中,类中的两个方法都被内置锁synchronized修饰,给第一部手机充电的同时(firstIPhoneByElectric方法中)给第二部手机充电(调用了secondIPhoneByElectric方法)。
因为充电线时多用的(synchronized是可重入的),所以同一个人给第二部手机充电时可以直接用该多用充电线进行充电(在同一线程在调用secondIPhoneByElectric时可以直接获得
当前对象的锁,进入secondIPhoneByElectric进行操作)。
设想:该充电线是单用的(如果synchronized不是可重入的),那么给第二部手机充电时必须等到第一部手机充电完成释放充电线后才能进行(当前线程在调用secondIPhoneByElectric
之前必须将firstIPhoneByElectric获取的锁释放掉),实际上该充电线已经被第一部手机占用,且无法释放,(该对象锁被当前线程所持有,且无法释放),所以此时会出现死锁。
这是什么原理呢?
当线程尝试获取锁时,可重入锁先尝试获取并更新status值,如果status == 0表示没有其他线程在执行同步代码,则把status置为1,当前线程开始执行。如果status != 0,
则判断当前线程是否是获取到这个锁的线程,如果是的话执行status+1,且当前线程可以再次获取锁。而非可重入锁是直接去获取并尝试更新当前status的值,如果status != 0
的话会导致其获取锁失败,当前线程阻塞。
释放锁时,可重入锁同样先获取当前status的值,在当前线程是持有锁的线程的前提下。如果status-1 == 0,则表示当前线程所有重复获取锁的操作都已经执行完毕,
然后该线程才会真正释放锁。而非可重入锁则是在确定当前线程是持有锁的线程之后,直接将status置为0,将锁释放。
原文地址:https://www.cnblogs.com/java-spring/p/11532109.html
- 结合Scikit-learn介绍几种常用的特征选择方法(下)
- Spring Cloud Bus之RabbitMQ初窥
- 结合Scikit-learn介绍几种常用的特征选择方法(上)
- 判断图片是否加载完成
- Spring Cloud Config客户端配置细节
- Spring Cloud Config服务端配置细节(二)之加密解密
- Redis3 cluster 集群配置测试
- Spring Cloud Config服务端配置细节(一)
- 分布式配置中心Spring Cloud Config初窥
- Git标签管理
- Docker容器如何共享数据
- Git工作区储藏兼谈分支管理中的一个小问题
- 轻量级容器Docker - 创建nginx容器
- Git关联远程仓库
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- three.js 制作逻辑转体游戏(上)
- three.js 制作魔方
- 安卓日常开发和逆向中常用的shell命令与非shell命令
- 使用python生成Latex公式语法
- 设计模式~桥梁模式
- 设计模式~模板方法模式
- three.js 郭先生制作太阳系
- 视频监控系统视频上云解决方案EasyCVR集成海康EHome私有协议系列——设备录像回看说明
- 视频监控系统视频上云解决方案EasyCVR集成海康EHome私有协议系列——报警模块说明
- three.js 制作一个三维的推箱子游戏
- IP摄像头RTSP协议网页无插件直播视频平台EasyNVR如何在直播或录像中添加定制化音频?
- three.js之初探骨骼动画
- three.js 自制骨骼动画(一)
- vue 人脸识别 demo
- 最完整的Explain总结,妈妈再也不担心我的SQL优化了