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

JavaEE初阶第七期:解锁多线程,从 “单车道” 到 “高速公路” 的编程升级(五)

专栏:JavaEE初阶起飞计划

个人主页:手握风云

一、死锁

1.1. 死锁的概念

        死锁是指两个或多个并发进程(或线程)在执行过程中,因争夺资源而造成的一种互相等待的现象。如果没有外力作用,这些进程将永远无法继续向前推进。

1.2. 造成死锁的原因

  • 同一把锁连续加锁两次

        由于synchronized具有可重入性,对于这种情况可以有效处理,但无法处理其他情况。

  • 两个线程两把锁,每个线程都先获取一把锁,再尝试获取对方的锁

        线程t1先拿到了locker1,线程t2也获取到了locker2。然后线程t1尝试获取locker2,线程t2尝试获取locker1。就如同把房门钥匙锁在车里面,而车钥匙又锁在家里面,就这样也构成了死锁。

public class Demo1 {public static void main(String[] args) throws InterruptedException {Object locker1 = new Object();Object locker2 = new Object();Thread t1 = new Thread(() -> {synchronized (locker1) {System.out.println("t1拿到了locker1");// 此处的sleep目的是让t1和t2分别获得对应的locker1和locker2// 如果t1和t2同时获得对应的locker1和locker2,那么t1和t2会互相等待对方释放锁,造成死锁try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (locker2) {System.out.println("t1拿到了locker2");}}});Thread t2 = new Thread(() -> {synchronized (locker2) {System.out.println("t2拿到了locker2");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (locker1) {System.out.println("t2拿到了locker1");}}});t1.start();t2.start();t1.join();t2.join();}
}

        通过线程更直观的观察,我们会发现两个线程都各自卡在了尝试获取对方锁的过程中。如果没有人工干预的情况下,那么两个线程将会永远卡住。

  • N个线程M把锁

        在计算机科学中,有一个经典的并发编程问题——哲学家就餐问题:五位哲学家围坐圆桌,桌上有五碗意大利面和五把餐叉,每位哲学家两侧各有一把餐叉,哲学家只能用两侧餐叉吃面,且吃面与思考交替进行。当所有哲学家同时拿起左侧的叉子时,由于右侧叉子在别的哲学家手里,每个人就需要等待放下别人手里的叉子,此时就会死锁。

1.3. 如何避免出现死锁

        线程一旦出现死锁,线程就会卡死了,后序的逻辑也无法正常执行了,从而产生了bug。而由于死锁的出现是概率性的,虽然概率小,但是也需要重视。

  • 造成死锁的必要条件
  1. 互斥性:如果一个线程已经占用了一个资源,其他线程就不能再申请该资源,直到该资源被释放,这是死锁发生的基础。
  2. 锁不可被抢占:一旦一个线程获得了一把锁之后,它就一直拥有这把锁,其他线程要想获得,只能阻塞等待。
  3. 请求与保持:一个线程在拿到一把锁时,第一把锁还没释放,又去请求第二把锁。
  4. 循环等待:等待锁释放的条件顺序构成了循环。

        只有当以上四个条件同时满足时,才可能发生死锁。 只要破坏其中任何一个条件,就可以有效预防死锁的发生。

public class Demo2 {public static void main(String[] args) throws InterruptedException {Object locker1 = new Object();Object locker2 = new Object();Thread t1 = new Thread(() -> {synchronized (locker1) {System.out.println("t1拿到了locker1");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}// locker1释放了,t2可以拿到synchronized (locker2) {System.out.println("t1拿到了locker2");}});Thread t2 = new Thread(() -> {synchronized (locker2) {System.out.println("t2拿到了locker2");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (locker1) {System.out.println("t2拿到了locker1");}}});t1.start();t2.start();t1.join();t2.join();}
}

二、volatile关键字

2.1. 内存可见性引起的线程安全问题

import java.util.Scanner;public class Demo3 {private static int flag = 0;public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(() -> {while (flag == 0) {}System.out.println("t1线程结束");});Thread t2 = new Thread(() -> {Scanner in = new Scanner(System.in);System.out.println("请输入flag的值:");flag = in.nextInt();System.out.println("t2线程结束");});t1.start();t2.start();t1.join();t2.join();}
}

        当我们输入一个非零值时,线程2结束了,但线程1并没有正确结束。这个Bug产生的原因就是内存的可见性。t2线程中flag变量的修改,但对于t1线程“不可见了”。

        对于Java编程语言的设计者来说,考虑到一个问题:写代码的程序员的水平参差不齐。虽然有的程序员水平不高,写代码效率较低。编译器在编辑执行的时候,分析理解现有代码的意图和效果,然后自动对整个代码进行优化和调整,在确保程序执行逻辑不变的前提下,进而提高效率。但在某些特定场景下,编译器会出现误判。

        对于上面的代码来说,编译器看到的是有一个flag会快速反复读取这个内存的值。因为反复执行(读取、比较……),每次拿到的flag值是一样的,读取内存的操作相比读取寄存器会耗时很多,于是编译器就会把从内存中读取flag的操作优化掉,直接从寄存器中读取。但在t2线程中对变量flag进行了修改,编译器也不能确定t2线程里的flag到底能不能执行到,以及啥时候执行。

2.2. volatile

        通过这个关键字,提醒编译器,某个变量是“易变的”,此时就不要针对这个变量进行上述优化。

import java.util.Scanner;public class Demo3 {// 加上volatile关键字之后private static volatile int flag = 0;public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(() -> {while (flag == 0) {}System.out.println("t1线程结束");});Thread t2 = new Thread(() -> {Scanner in = new Scanner(System.in);System.out.println("请输入flag的值:");flag = in.nextInt();System.out.println("t2线程结束");});t1.start();t2.start();t1.join();t2.join();}
}

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

相关文章:

  • ElasticSearch快速入门-1
  • MSPM0G3507学习笔记(一) 重置版:适配逐飞库的ti板环境配置
  • 服装零售企业跨区域运营难题破解方案
  • 如何将大型视频文件从 iPhone 传输到 PC
  • PoE 延长器——让网络部署更自由
  • 第十章:HIL-SERL 真实机器人训练实战
  • Docker拉取bladex 、 sentinel-dashboard
  • 【阿里巴巴JAVA开发手册】IDE的text file encoding设置为UTF-8; IDE中文件的换行符使用Unix格式,不要使用Windows格式。
  • Android BitmapRegionDecoder 详解
  • Java启动脚本
  • vue create 和npm init 创建项目对比
  • error MSB8041: 此项目需要 MFC 库。从 Visual Studio 安装程序(单个组件选项卡)为正在使用的任何工具集和体系结构安装它们。
  • React 渲染深度解密:从 JSX 到 DOM 的初次与重渲染全流程
  • 最快实现的前端灰度方案
  • 因果语言模型、自回归语言模型、仅解码器语言模型都是同一类模型
  • 同步(Synchronization)和互斥(Mutual Exclusion)关系
  • 【机器人】复现 DOV-SG 机器人导航 | 动态开放词汇 | 3D 场景图
  • (超详细)数据库项目初体验:使用C语言连接数据库完成短地址服务(本地运行版)
  • 敏捷开发在国际化团队管理中的落地
  • 二维码驱动的独立站视频集成方案
  • 碰一碰发视频源码搭建与定制化开发:支持OEM
  • 译码器Multisim电路仿真汇总——硬件工程师笔记
  • TensorFlow 安装使用教程
  • MySQL数据库----DML语句
  • 【2.4 漫画SpringBoot实战】
  • 【模糊集合】示例
  • vue-37(模拟依赖项进行隔离测试)
  • Unity Android与iOS自动重启
  • uniapp打包微信小程序主包过大问题_uniapp 微信小程序时主包太大和vendor.js过大
  • 《导引系统原理》-西北工业大学-周军-“2️⃣导引头的角度稳定系统”