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

JavaEE----多线程(二)

文章目录

  • 1.进程的状态
  • 2.线程的安全引入
  • 3.线程安全的问题产生原因
  • 4.synchronized关键字的引入
    • 4.1修饰代码块
    • 4.2修饰实例方法
    • 4.3修饰静态方法
    • 4.4对象头介绍
    • 4.5死锁-可重入的特性
  • 5.关于死锁的分析总结
    • 5.1死锁的分析
    • 5.2死锁成因的必要条件
    • 5.3死锁的解决方案

1.进程的状态

public class Test {public static void main(String[] args) throws InterruptedException {Thread t=new Thread(()->{});System.out.println(t.getState());//线程已经创建,但是没有开始执行,这个时候的状态就是new状态t.start();t.join();//TERMINATED就是线程结束之后的这个线程的状态:terminatedSystem.out.println(t.getState());}}

下面的这个就是在我们的t线程里面设计一个死循环,这个时候我们就可以使用getstate获取到这个时候的状态就是我们的runnable状态的;

image-20241021191206127

下面的这个是对于timed-waiting状态的演示:

image-20241021191820497

2.线程的安全引入

线程的安全问题:主要就是这个线程调度的随机性;

下面的这个里面,我们是对于这个全局的静态变量是count=0,我们在这个主方法里面对于这个count分别加上50000次,这个时候我们正常情况下结果应该是100000,但是这个打印结果不是100000,主要就是因为这个调度器的执行问题导致的这个结果不是100000;

实际上,这个count进行++的时候,需要经过三个步骤,分别是

load:把内存里面的数据进行读取到CPU寄存器里面去;

add:把这个寄存器里面的数据count++;

save:把这个寄存器里面的计算之后的数据放到这个内存里面去;

image-20241021205334027

为什么计算之后的这个结果不是100000呢,下面我们画一下这个计算的过程:

如果是下面的这个情况,我们的两个线程的三步骤都是连续的,这个时候我们的count就会被加上去,这个时候就是2;

image-20241021210838886

但是如果下面的这个情况,就是两个线程之间的这个三步操作就会失效,只有一个会发挥作用,因为其中的一个过程被中断了:

image-20241021211040128

3.线程安全的问题产生原因

1.操作系统里面对于线程的调度是随机的(抢占式执行);

​ 我们想要解决问题肯定不可以从这个入手,因为这个是取决于我们的操作系统,我们很难对于这个抢占 式执行的现状进行控制;

2.两个线程,针对于一个变量进行修改;

​ 上面的这个就是属于这个情况,两个线程都是针对于这个count进行操作,因此这个时候因为这个调度执 行的步骤可能会被切断,因此这个时候就会出现问题;

3.修改操作不是原子的;

​ 什么是原子:上面的这个count++就不是原子的,原子的简单的就可以理解为这个操作的步骤是一步到位

​ 还是需要分为多次进行执行,上面的这个load,add,save需要分为三个步骤进行执行,因此这个就不是原 子的;

​ 假设我们的这个步骤一步就可以完成,这个时候我们就把这个操作叫做原子的;

4.内存可见性问题;

5指令重排序问题;(4,5)我们暂时没有遇到,因此这个地方不进行过多的介绍;

4.synchronized关键字的引入

4.1修饰代码块

我们的这个synchronized实际上就是对于这个操作进行加锁的操作,只有我们的一个线程的三个步骤全部执行完毕之后,我们的另外一个线程才会被执行,相当于我们的t1对于这个全局的变量++的时候,这个就是出于上锁的状态,其他的线程无法进行操作,只有当我们的这个线程执行完毕之后,这个锁被释放掉,也就是开锁,这个时候我们的其他的线程才可以继续执行;

这样的操作保证了这个不同的线程之间的这个操作的独立性,就不会出现上面介绍的一个线程的三个步骤被另外一个线程打断,出现两个线程的操作交叉执行的问题;就是这个load,add,save就是各走各的,而且三个过程是连续的,不会被中断;

image-20241021205544049

除了上面的这个synchronized修饰代码块之外,我们的这个synchronized还是可以修饰我们的静态方法和我们的实例方法的:

4.2修饰实例方法

所谓的这个实例方法,其实就是为了和我们的静态方法进行区分,就是一个类里面的普通的成员方法,下面的这个就是我们的synchronized修饰我们的实例方法,下面的两个本质是等效的,因为这个synchronized修饰我们的实例方法本质上就是对于这个this锁对象进行操作,这个时候的锁对象就是我们的this;

因此这样来讲,上面的操作和我们的下面的这个修饰方法就是一样的,只不过我们的这个下面的写法里面,把这个锁对象隐藏了起来;

image-20241021215816397

4.3修饰静态方法

下面的两个写法就是一样的,就是我们的synchronized关键字修饰我们的静态成员方法,相当于这个代码块里面的这个参数就是我们的类对象;

我们的代码里面定义了一个类,那么这个里面就一定会有一个类对象,而且一个类只会有一个类对象,不会有多个的;

public class Test {public static void main(String[] args) {synchronized public static void increase(){}public static increase2(){synchronized (counter.class){}}}
}

4.4对象头介绍

synchrinozed修饰的这个锁是存在于我们的这个对象头里面的,那么什么是对象头:

对象头就是我们进行这个对象的创建的时候,一个对象会有自己的内存空间,在这个内存空间里面,除了我们自己对于这个对象定义的属性,这个对象还会有些默认的属性,这个默认的属性就是在我们的对象头里面的;

在对象头里面,就有属性是存放说明我们的这个对象是不是加上了锁的;

4.5死锁-可重入的特性

什么是可重入 ,就是对于一个对象,我们连续加锁,这个时候不会出现死锁的情况;

我们使用下面的这个案例对于死锁进行说明:

死锁就是像下面的这个情况一样,我们连续对哦与一个锁对象多次加锁,这个时候就会出现死锁,具体的讲就是线程被卡死了;

synchronized(locker){synchronized(locker){.......}
}

为什么会出现下面的这个死锁的情况,就是我们的第一次加锁的时候,我们的第二次操作正常情况下是进不去的,需要第一次的这个吧这个锁打开之后我们才可以第二次进入,但是下面的这个情况下我们的这个锁想要打开,只有等到这个操作执行完,就是这个代码块执行完,也就是执行到我们下面的这个示例代码的第二个}位置才可以,但是想要执行到这个第二个}位置,必须要执行这个第二次的加锁的操作,这个就是矛盾的地方;

因此这个时候想要加锁,但是这个执行又无法结束,因此这个时候就会出现线程卡死的情况,也就是我们说的死锁现象,为了处理这个问题,synchronized关键字引入了这个可重入的特性,就是对于这个相同的锁对象,我们可以重入,就是反复的入,也就是反复地加锁,这个是可以被允许的;

当然,这个死锁的现象是针对于这个相同的锁对象多次加锁,这个时候才可能会出现死锁的情况,如果每一次加锁针对的锁对象不是一样的,这个时候是不会出现我们的死锁现象的;

synchronized(locker){//下面的这个是针对于一个新的锁对象进行加锁,这个时候肯定不会出现死锁的情况,无论是不是可重入的synchronized(locker2){.......}
}

那么,在这个可重入的特性下,我们的这个锁什么时候打开呢,正确答案是,直到所有的这个锁全部加上之后,直到我们的这个最外层大括号的时候,这个锁才会被打开;

具体到下面的这个情况,就是执行到最后一个}的时候,这个锁才会被打开,这个过程里面,我们会不断的进行计数,就是这个锁一共加上了几层,即n++,打开的时候,也会不断的对于这个n–直到这个走到最后一个}的时候,这个时候的n=0,也就是我们释放锁的时候;

synchronized(locker){synchronized(locker){synchronized(locker){synchronized(locker){synchronized(locker){..........}}}}
}

5.关于死锁的分析总结

5.1死锁的分析

1.一个对象,被连续两次上锁,这个时候如果是不可重入锁,就会发生死锁的现象;

2.两个对象,两把锁,这个时候无论是不是可重入的,都会发生这个死锁现象;

这个经典案例就是我们的钥匙落在了车里,车钥匙落在了家里,这个时候就会出现思索的现象;

这个时候家和车就是两个对象,我们的车钥匙和家钥匙就是锁,这个时候出现的情况就是我们的死锁的现象;

3.N个对象,M把锁,这个时候就是上面的两个对象两把锁的扩展,这个时候更加容易出现死锁的现象;

最经典的N歌对象,M把锁的问题就是我们的哲学家就餐问题:

这个时候我们是使用5个滑稽作为案例的,没有画出来很多,我们的滑稽在就餐的时候,每一个人都是拿走的自己的最近的一个筷子,这个时候,每一个人只有一个,谁都无法就餐,所有的人就是阻塞的状态,这个时候就会出现死锁的现象;(下面我们会介绍这个解决的方案);

image-20241022183258980

5.2死锁成因的必要条件

死锁的成因,需要满足下面的这四个条件,并且是同时满足的:

1.互斥使用(死锁的基本特性):当一个线程有一把锁之后,另外一个线程想要使用这个锁,需要进行阻塞等待;

2.不可抢占(死锁的基本特性):当一个锁被线程1拿到之后,线程2只能等待这个线程1主动地进行释放,否则只能处于等待的状态;

3.请求保持(代码结构):一个线程尝试获取多把锁,先拿到第一把锁之后,尝试获取第二把锁,获取这个锁的时候,第一把锁不会被释放;

4.循环等待/环路等待:等待之间的依赖关系,形成了换;

例如一个例子:钥匙锁在了车里,车钥匙锁在了家里;这个就是一个循环的环路等待,这个结果就是一个死循环,也是死锁的一个成因;

5.3死锁的解决方案

解决死锁问题的核心就是要破坏上面的必要条件,但是这个里面的第一和第二个必要条件就是我们的锁的特性,因此这个不需要进行考虑,我们主要针对于3,4两个必要条件进行解决;

针对于3这个现象,我们需要进行这个代码结构的调整,不要把两个加锁的代码放到一个代码块里面去;

针对于4这个现象,我们需要进行编号操作,可以有效的解决这个问题:

还是使用这个哲学家的就餐问题,我们进行编号之后,让每一个滑稽取出来这个最小的编号的筷子(自己面前的两个筷子里面的最小的),这样的话,我们的问题就解决了;

image-20241022184019309

我们可以分析一下这个就餐的过程,我们的拿筷子的情况如图所示,每一个人拿的都是自己的这个面前的两个里面的最小的编号,我们的最后一个滑稽取筷子的时候,1和5相比,肯定是1小,这个时候他就不可以取走这个5编号的筷子,这个时候的1已经是被和他相邻的这个滑稽取走了,因此这个时候,只能等待人家用完;

因此这个时候我们的左上角的这个滑稽就可以拿到这个5开始就餐,放下筷子之后,我们的左下角的这个滑稽拿到这个4号筷子吃饭,以此类推,直到我们的右上角的这个滑稽放下筷子,这个时候我们的最上面的这个滑稽就可以吃饭了,这个线程的死锁问题就被解决了;

image-20241022184407529
定是1小,这个时候他就不可以取走这个5编号的筷子,这个时候的1已经是被和他相邻的这个滑稽取走了,因此这个时候,只能等待人家用完;

因此这个时候我们的左上角的这个滑稽就可以拿到这个5开始就餐,放下筷子之后,我们的左下角的这个滑稽拿到这个4号筷子吃饭,以此类推,直到我们的右上角的这个滑稽放下筷子,这个时候我们的最上面的这个滑稽就可以吃饭了,这个线程的死锁问题就被解决了;

相关文章:

JavaEE----多线程(二)

文章目录 1.进程的状态2.线程的安全引入3.线程安全的问题产生原因4.synchronized关键字的引入4.1修饰代码块4.2修饰实例方法4.3修饰静态方法4.4对象头介绍4.5死锁-可重入的特性 5.关于死锁的分析总结5.1死锁的分析5.2死锁成因的必要条件5.3死锁的解决方案 1.进程的状态 public…...

【K8S】快速入门Kubernetes

之前企业都是使用容器化和来构建自己的服务和应用程序,其中容器化优点有很多:提升了部署效率、稳定性、提高了资源的利用率降低了成本。 但是也带来了一些新的问题:容器的数量变得很多,管理就是一个新的问题。所以Kubernetes就出…...

如何在 MySQL 中处理大量的 DELETE 操作??

全文目录: 开篇语前言摘要简介概述DELETE 操作的基本概念常用的 DELETE 方法 核心源码解读简单 DELETE 语句批量 DELETE 示例 案例分析案例1:使用简单 DELETE 删除用户数据案例2:使用分批 DELETE 应用场景演示场景1:用户管理系统场…...

LabVIEW中句柄与引用

在LabVIEW中,句柄(Handle) 是一种用于引用特定资源或对象的标识符。它类似于指针,允许程序在内存中管理和操作复杂的资源,而不需要直接访问资源本身。句柄用于管理动态分配的资源,如队列、文件、网络连接、…...

【三十四】【QT开发应用】音量图标以及滑动条,没有代码补全的小技巧

效果展示 鼠标位于音量图标区域内,显示出滑动条。鼠标移出音量图标区域内滑动条隐藏。鼠标点击音量图标,如果此时音量为0,音量变成50,如果此时音量不为零,音量变为0。 CVolumeButton.h 音量图标头文件 #pragma once …...

Android修改第三方应用相机方向

以下修改基于Android7.1 diff --git a/frameworks/base/core/java/android/hardware/Camera.java b/frameworks/base/core/java/android/hardware/Camera.java index 8c7434b..7201481 100755 --- a/frameworks/base/core/java/android/hardware/Camera.java b/frameworks/ba…...

Python 读取文件汇总

readline和readlines的区别 使用 open()读取文件时,readline是读取文件的一行;而readlines是加载全部文档,以list形式保存每一行内容。 使用with避免资源泄露 with语句不仅限于open()函数,任何实现了上下文管理协议的对象都可以…...

云原生:一张图了解devops 中CI/CD

一个典型的云原生应用的开发和部署过程,其中涉及到的主要工具有 Git、Docker、Jenkins/CircleCI、Ansible、Kubernetes 等。以下是每个步骤的简要说明: 开发人员(Developers)使用 Git 进行版本控制,他们将代码推送到 G…...

无人机之自组网通信技术篇

无人机的自组网通信技术是一种利用无人机作为节点,通过无线通信技术实现节点间自主组网、动态路由和数据传输的技术。 一、技术原理与特点 技术原理:无人机自组网技术基于自组织网络(Ad-Hoc Network)的原理,通过无线…...

【WebLogic】Oracle发布2024年第四季度中间件安全公告

Oracle于美国时间2024年10月15日发布了 WebLogic 12c(12.2.1.4.0)和14c(14.1.1.0.0)两个大版本2024年第4季度的安全公告,涉及漏洞ID共计 6 个,包含2个高危漏洞 2 个,4个中危漏洞,其中…...

Java集合(3:Set和Map)

文章目录 Set概述哈希值HashSet去重原理LinkedHashSetTreeSet自定义排序规则 Map概述Map的基本方法Map集合的获取功能哈希表HashMap底层源码 特点注意 Set 概述 Set集合也是一个接口,继承自Collection,与List类似,都需要通过实现类来进行操…...

【Golang】Gin框架中如何定义路由

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,…...

CPU内存飙升

CPU 飙升介绍 CPU 飙升是指中央处理器(CPU)的使用率在短时间内急剧上升,达到一个较高的水平。正常情况下,CPU 会根据系统和应用程序的需求合理分配资源,使用率会在一定范围内波动。但当 CPU 飙升时,可能会导…...

【Java】LinkedList实现类的使用

LinkedList实现类的使用 package com.star.test04;import java.util.Iterator;import java.util.LinkedList;/** * author : Starshine */public class Test { //这是main方法,程序的入口 public static void main(String[] args) { /* LinkedL…...

创建人物状态栏

接下来,我们来尝试制作一下我们的UI,我们会学习unity基本的UI系统 ************************************************************************************************************** 我们要先安装一个好用的插件到我们的unity当中,帮助…...

django5入门【01】环境配置

注意: ⭐前提:安装了annaconda(python版本管理工具),如果没有安装,强烈建议安装一下!!!操作: 前言: 这里新创建一个名为“python_3.11_start_dja…...

1000集《楼兰》系列短剧开机仪式在疆举行,开启全球传播新篇章

2024年10月18日,光明媚,秋风送爽。 在这个收获的季节里,倍受期待的楼兰系列短剧《楼兰之天女归来》和《楼兰之时空秘宝》在新疆吐鲁番东方红卓览文化博物馆举行了隆重的开机仪式,正式拉开了摄制的序幕。 1000集《楼兰》系列短剧…...

【景观生态学实验】实验五 景观生态脆弱性评价

实验目的 1.学习层次分析模型思路,对丹江口库区2000年景观生态脆弱性评价建模:通过实验课的学习,深入理解层次分析(Analytic Hierarchy Process,AHP)理论与模型,了解其在决策问题中的应用&…...

ChatGPT 现已登陆 Windows 平台

今天,OpenAI 宣布其人工智能聊天机器人平台 ChatGPT 已开始预览专用 Windows 应用程序。OpenAI 表示,该应用目前仅适用于 ChatGPT Plus、Team、Enterprise 和 Edu 用户,是一个早期版本,将在今年晚些时候推出"完整体验"。…...

和鲸社区数据科学实训季,西安交通大学圆满收官,西安,后会有期!

和鲸社区数据科学实训季活动已走进数十家高校,在西安的收官之站,落定西安交通大学管理学院,为本次西安之旅画上了圆满的句号。 和鲸社区 2024秋 数据科学实训季以“帮助同学积累真实场景项目经验”为出发点,提供 60 个数据科学实践…...

Python|GIF 解析与构建(5):手搓截屏和帧率控制

目录 Python|GIF 解析与构建(5):手搓截屏和帧率控制 一、引言 二、技术实现:手搓截屏模块 2.1 核心原理 2.2 代码解析:ScreenshotData类 2.2.1 截图函数:capture_screen 三、技术实现&…...

如何在看板中体现优先级变化

在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...

【android bluetooth 框架分析 04】【bt-framework 层详解 1】【BluetoothProperties介绍】

1. BluetoothProperties介绍 libsysprop/srcs/android/sysprop/BluetoothProperties.sysprop BluetoothProperties.sysprop 是 Android AOSP 中的一种 系统属性定义文件(System Property Definition File),用于声明和管理 Bluetooth 模块相…...

聊一聊接口测试的意义有哪些?

目录 一、隔离性 & 早期测试 二、保障系统集成质量 三、验证业务逻辑的核心层 四、提升测试效率与覆盖度 五、系统稳定性的守护者 六、驱动团队协作与契约管理 七、性能与扩展性的前置评估 八、持续交付的核心支撑 接口测试的意义可以从四个维度展开,首…...

如何在最短时间内提升打ctf(web)的水平?

刚刚刷完2遍 bugku 的 web 题,前来答题。 每个人对刷题理解是不同,有的人是看了writeup就等于刷了,有的人是收藏了writeup就等于刷了,有的人是跟着writeup做了一遍就等于刷了,还有的人是独立思考做了一遍就等于刷了。…...

AI,如何重构理解、匹配与决策?

AI 时代,我们如何理解消费? 作者|王彬 封面|Unplash 人们通过信息理解世界。 曾几何时,PC 与移动互联网重塑了人们的购物路径:信息变得唾手可得,商品决策变得高度依赖内容。 但 AI 时代的来…...

HarmonyOS运动开发:如何用mpchart绘制运动配速图表

##鸿蒙核心技术##运动开发##Sensor Service Kit(传感器服务)# 前言 在运动类应用中,运动数据的可视化是提升用户体验的重要环节。通过直观的图表展示运动过程中的关键数据,如配速、距离、卡路里消耗等,用户可以更清晰…...

Webpack性能优化:构建速度与体积优化策略

一、构建速度优化 1、​​升级Webpack和Node.js​​ ​​优化效果​​:Webpack 4比Webpack 3构建时间降低60%-98%。​​原因​​: V8引擎优化(for of替代forEach、Map/Set替代Object)。默认使用更快的md4哈希算法。AST直接从Loa…...

[大语言模型]在个人电脑上部署ollama 并进行管理,最后配置AI程序开发助手.

ollama官网: 下载 https://ollama.com/ 安装 查看可以使用的模型 https://ollama.com/search 例如 https://ollama.com/library/deepseek-r1/tags # deepseek-r1:7bollama pull deepseek-r1:7b改token数量为409622 16384 ollama命令说明 ollama serve #&#xff1a…...

关于uniapp展示PDF的解决方案

在 UniApp 的 H5 环境中使用 pdf-vue3 组件可以实现完整的 PDF 预览功能。以下是详细实现步骤和注意事项&#xff1a; 一、安装依赖 安装 pdf-vue3 和 PDF.js 核心库&#xff1a; npm install pdf-vue3 pdfjs-dist二、基本使用示例 <template><view class"con…...