当前位置: 首页 > news >正文

JUC之锁

公平锁、非公平锁

        公平锁指的是多线程按照申请锁的顺序来获取锁;

非公平锁指的是多线程不按照申请锁的顺序来获取锁。可能会出现优先级反转(后者居上)

公平锁为了保证线程申请顺序,势必要付出一定的性能代价,因此其吞吐量一般低于非公平锁

synchronized 支持非公平锁;Lock默认非公平锁,但支持公平锁

可重入锁(递归锁)

可重入锁又名递归锁,是指 同一个线程在外层方法获取了锁,在进入内层方法会自动获取锁。

synchronized void setA() throws Exception{Thread.sleep(1000);setB();
}synchronized void setB() throws Exception{Thread.sleep(1000);
}

如果使用的锁不是可重入锁的话,setB()可能不会被当前线程执行,从而造成死锁。可重入锁可以在一定程度上避免死锁。

自旋锁

自旋锁是指尝试获取锁的线程不会立即阻塞,而是采用循环的方式去尝试获取锁

自旋等待的时间必须要有一定的限度,如果自旋超过了限定次数,就应当挂起线程。自旋锁的实现原理是CAS

乐观锁、悲观锁

        乐观锁与悲观锁不是指具体的什么类型的锁,而是处理并发同步的策略。

悲观锁 - 悲观锁对于并发采取悲观的态度,认为:不加锁的并发操作一定会出问题。悲观锁适合写操作频繁的场景;

乐观锁 - 乐观锁对于并发采取乐观的态度,认为:不加锁的并发操作也没什么问题。对于同一个数据的并发操作,是不会发生修改的。在更新数据的时候,会采用不断尝试更新的方式更新数据。乐观锁适合读多写少的场景;

独享锁与共享锁

        独享锁与共享锁是一种广义上的说法,从实际用途上来看,也常被称为互斥锁与读写锁

独享锁 - 独享锁是指 锁一次只能被一个线程所持有

共享锁 - 共享锁是指 锁可被多个线程所持有

无锁 、 偏向锁、轻量级锁、重量级锁

一个对象其实有四种锁状态,它们级别由低到高;

无锁:没有对资源进行锁定,任何线程都可以尝试去修改它,但同时只有一个线程能修改成功;

偏向锁:是指一段同步代码一直被一个线程所访问,那么该线程会自动获取锁,降低获取锁的代价;

轻量级锁:当锁是偏向锁的时候,被另一个线程所访问,偏向锁就会升级为轻量级锁,其他线程会通过自旋的形式尝试获取锁,不会阻塞,提高性能;

重量级锁:若当前只有一个等待线程,则该线程通过自旋进行等待。但是当自旋超过一定的次数,或者一个线程在持有锁,一个线程在自旋,又有第三个线程来访时,轻量级锁升级为重量级锁;升级为重量级锁时,等待锁的线程都会进入阻塞状态;

死锁

两个或多个线程因竞争资源出现等待彼此造成永久阻塞的情况;

产生条件:

        1.一段时间内某资源仅为一线程所占用

        2.线程对已获得的资源保持不放

        3.进程已获得的资源在未使用完之前,不能剥夺

        4.线程之间出现循环等待资源

public class DeadLock {static Object object1 = new Object();static Object object2 = new Object();public static void main(String[] args) {new Thread(() -> {synchronized (object1){System.out.println("a线程已经获得锁1 --> 等待获得锁2");try {TimeUnit.SECONDS.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}synchronized (object2){System.out.println("a线程成功获得锁2");}}},"a").start();new Thread(() -> {synchronized (object2){System.out.println("b线程已经获得锁2 --> 等待获得锁1");try {TimeUnit.SECONDS.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}synchronized (object1){System.out.println("b线程成功获得锁1");}}},"b").start();}
}

解决办法:

        1.让程序每次至多只能获得一个锁

        2.设计时考虑清楚锁的顺序,尽量减少嵌在的加锁交互数量

        3.死锁检测

        4.设置加锁时限

http://www.lryc.cn/news/68488.html

相关文章:

  • C++中的 cout 和 printf 用法
  • Maven基础使用
  • 【C++ 入坑指南】(06)运算符
  • 了解一下js中的函数式编程
  • 动态HTTP代理在linux里的使用
  • 软考证书值得考吗?怎么考?
  • 超级秘密文件夹忘记密码的解决办法
  • 脑的物理系统
  • 1054. 距离相等的条形码(leetcode,堆问题,priority_queue)-------------------c++实现
  • QT开发实战-动态壁纸软件
  • Netty核心组件模块(一)
  • Robot Framework+Jenkins持续集成UI自动化项目
  • 【ROS】ROS1编程速览
  • 探索智能化:TOOM解析未来稿件校验系统的技术进展与应用展望
  • Java程序员从青铜到王者,不同段位的薪资和技能变化
  • tinyWebServer 学习笔记——二、HTTP 连接处理
  • 深入浅析Linux Perf 性能分析工具及火焰图
  • java关键术语
  • 1. 两数之和【简单】
  • 《编码——隐匿在计算机软硬件背后的语言》精炼——第17章(自动操作)
  • 用Colab免费部署AI绘画云平台Stable Diffusion webUI
  • R.I.P,又一位程序员巨佬——左耳朵耗子陨落
  • 捷威信keithley吉时利2410数字源表 销售回收KEITHLEY2470新款源表
  • 第二十九回:如何给ListView添加分隔线
  • 用友 LRP计划维护视图
  • 数组--part 5--螺旋矩阵(力扣59/54)(剑指offer 29)
  • 加密解密软件VMProtect入门使用教程(九)许可制度之许可系统功能
  • MySQL基础-事务详解
  • python 读写csv文件方法
  • 命令行更新Windows