【多线程】线程安全问题和解决方案

news2025/1/15 22:49:48

我们来看下面这一段代码

public class demo {
    public static void main(String[] args) throws InterruptedException {
        Cou count = new Cou();
        Thread t1 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                count.add();
            }
        });

        Thread t2 = new Thread(() -> {
            for (int i = 0; i < 10000; i++) {
                count.add();
            }
        });

        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(Cou.count);
    }
}

class Cou {
    static int count = 0;
    public void add() {
        count++;
    }
}

我们期望的结果是得到20000,但是实际上这个值是随机的,它一定是小于20000的。这就是线程安全带来的问题。

说到线程安全问题我们就要知道什么是原子性,我们举一个例子:我们把一段代码想象成一个房间,每个线程就是要进入这个房间的人。如果没有任何机制保证,A进入房间之后,还没有出来;B 是不是也可以进入房间,打断 A 在房间里的隐私。这个就是不具备原子性的。那我们应该如何解决这个问题呢?是不是只要给房间加一把锁,A 进去就把门锁上,其他人是不是就进不来了。这样就保证了这段代码的原子性了。有时也把这个现象叫做同步互斥,表示操作是互相排斥的。一条Java指令不一定是原子的,也不一定只是一条指令。比如我们上面代码的count++,其实是由三步操作组成的:①把内存数据读取到cpu上 ②修改数据 ③把修改的数据存到cpu上。不保证原子性会给多线程带来很多问题,如果一共线程正在修改数据,这个时候另一个线程也开始操作相同点数据就会打断第一个线程的工作,这样结果就很有可能是有问题的。

可见性,就是一个线程修改变量能够及时被其他线程看到

我们来看下面的图
在这里插入图片描述

线程的调度是随机的,抢占式执行,这样就可能会导致执行顺序和逻辑出现问题,我们不知道有多少次自增是正确的,所以结果我们并不知道。

产生线程安全问题的原因

①操作系统中,线程调度的顺序是随机的,抢占式执行。

②两个线程针对同一个变量进行修改。

③修改操作不是原子的。

④内存可见性问题。

⑤指令重排序问题。

我们可以通过加锁来解决线程安全问题

谈到锁我们就要了解 synchronized 的特性,synchronized 会起到互斥效果,如果某个线程执行到某个对象的synchronized 中时其他线程也执行到同一个对象那么synchronized 就会阻塞等待

阻塞等待:针对每一把锁,操作系统内部都维护了一个等待队列。当这个锁被某个线程占有的时候,其他线程尝试进行加锁,就加不上了,就会阻塞等待,一直等到之前的线程解锁之后,由操作系统唤醒一个新的线程,再来获取到这个锁。

进入synchronized 修饰的代码块相当于加锁,退出synchronized 修饰的代码块相当于是解锁。

上一个线程解锁之后, 下一个线程并不是立即就能获取到锁. 而是要靠操作系统来 “唤醒”. 这也就是操作系统线程调度的一部分工作.假设有 A B C 三个线程, 线程 A 先获取到锁, 然后 B 尝试获取锁, 然后 C 再尝试获取锁, 此时 B和 C 都在阻塞队列中排队等待. 但是当 A 释放锁之后, 虽然 B 比 C 先来的, 但是 B 不一定就能获取到锁, 而是和 C 重新竞争, 并不遵守先来后到的规则 .

synchronized(lock){
    synchronized(lock){
        ......
    }//2
}//1

我们假设第一次加锁成功,这个时候lock就属于是被锁定状态,再进行第二次加锁,这个时候应该是阻塞等待状态,要想加锁成功就需要等到锁释放后才能加锁成功。但是实际上一旦第二次加锁阻塞了就会出现死锁。要想第二次加锁成功就需要第一次加锁释放锁,第一次要想释放锁就需要执行到 1 的的位置,而执行到 1 的位置就需要第二次加锁成功,但是由于第二次加锁导致了阻塞,这样就没有办法执行到 1 ,也就无法释放锁。

synchronized 是可重入锁,就是一个线程针对一把锁连续加锁两次,不出现死锁,这种锁就是可重入锁。可以有效的解决上面死锁的问题。

在可重入锁的内部,包含了 “线程持有者” 和 “计数器” 两个信息。如果某个线程加锁的时候,发现锁已经被人占用,但是恰好占用的正是自己,那么仍然可以继续获取到锁,并让计数器自增。解锁的时候计数器递减为 0 的时候,才真正释放锁。(才能被别的线程获取到 )

造成死锁的四个必要条件

1.互斥作用(锁的基本特性):当一个线程加锁了,另一个线程想要获取这把锁就需要阻塞等待。

2.不可抢占(锁的基本特性):当锁被一个线程拿到后,另一个线程只能等这个线程释放锁后才能拿到这把锁,不能强行抢占。

3.请求保持(代码结构):一个线程尝试获取多把锁(拿到一把锁后还没有释放就想着获取另一把锁)。

4.循环等待/环路等待(代码结构):等待的依赖关系形成了环。

只有同时出现上述四种情况才会出现死锁。我们要想解决死锁,就需要避免上述四种情况同时出现,第一条和第二条是synchronized的特性,我们改变不了,所以只能避免三和四同时出现。对于三来所,避免编写锁嵌套逻辑并不好使,所以我们可以针对四来解决死锁。我们可以约定加锁的顺序,避免循环等待,我们可以针对锁进行编号,比如加多把锁的时候,先加编号小的锁,再加编号大的锁(所有线程都遵守)

我们来看下面的代码

public class Thread1 {
    private static int isQuit=0;
    public static void main(String[] args) {
        Thread t1 = new Thread(()->{
           while (isQuit==0){
				//循环什么都不做
           }
            System.out.println("t1退出");
        });

        Thread t2 = new Thread(()->{
            System.out.print("请输入isQuit:");
            Scanner scanner = new Scanner(System.in);
            //输入不为0则t1线程结束
            isQuit = scanner.nextInt();
        });
        t1.start();
        t2.start();
    }
}

在这里插入图片描述

我们预期结果应该是输入不为0则t1线程结束但是我们输入 1 t1线程并没有结,我们通过jconsole可以看到t1线程的状态是RUNNABLE正在执行。

在这里插入图片描述

这也是一个线程安全问题,之前我们是两个线程同时修改一个变量,这次是一个线程读一个线程修改,这种情况也有可能会有问题。这就是由内存可见性引起的。因为我们上面说过,load把isQuit的值读取到寄存器中,让后通过cmp指令判断是否为0,因为这个循环速度很快短时间内会进行大量的load和cmp操作,此时,编译器发现进行了这么多次load操作结果都是一样的并且load操作还很费时间,一次load相当于上万次cmp,所以编译器就做了一个大胆的决定,只在第一次循环的时候读取内存后续都不读取,只从寄存器中读取isQuit,这是编译器的自我优化,编译器的初衷是提升程序效率,但是提高程序效率的前提是逻辑不变,但是此时修改isQuit的操作是另一个线程操作的,编译器不能正确判断,以为isQuit没有修改,所以就引起了bug。而编译器什么时候会优化我们无法推断,所以我们可以用volatile 来告诉编译器不要优化,这样程序就不会出错。

public class Thread1 {
    private volatile static int isQuit=0;
    public static void main(String[] args) {
        Thread t1 = new Thread(()->{
           while (isQuit==0){
				//循环什么都不做
           }
            System.out.println("t1退出");
        });

        Thread t2 = new Thread(()->{
            System.out.print("请输入isQuit:");
            Scanner scanner = new Scanner(System.in);
            //输入不为0则t1线程结束
            isQuit = scanner.nextInt();
        });
        t1.start();
        t2.start();
    }
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1111650.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

王道计算机考研 操作系统学习笔记篇章二: 进程管理

目录 进程与线程 进程的概念 概念 进程的组成 PCB 程序段、数据段 进程的特征 总结 进程的状态与转换 进程的状态 创建态、就绪态 运行态 阻塞态 终止态 进程的转换 进程的组织 链接方式 索引方式 总结 进程控制 什么是进程控制 如何实现进程控制 进程控制相关的原…

Opencv之RANSAC算法用于直线拟合及特征点集匹配详解

Opencv之RANSAC算法用于直线拟合及特征点集匹配详解 讲述Ransac拟合与最小二乘在曲线拟合上的优缺点 讲述在进行特征点匹配时&#xff0c;最近邻匹配与Ransac匹配的不同之处 另外&#xff0c;Ransac也被用于椭圆拟合、变换矩阵求解等 1. 直线拟合 1.1 原理 RANSAC(RANdom …

两分钟搞懂UiAutomator自动化测试框架

1. UiAutomator简介 UiAutomator是谷歌在Android4.1版本发布时推出的一款用Java编写的UI测试框架&#xff0c;基于Accessibility服务。其最大的特点就是可以跨进程操作&#xff0c;可以使用UiAutomator框架提供的一些方便的API来对安卓应用进行一系列的自动化测试操作&#xf…

Linux程序调试器——gdb的使用

gdb的概述 GDB 全称“GNU symbolic debugger”&#xff0c;从名称上不难看出&#xff0c;它诞生于 GNU 计划&#xff08;同时诞生的还有 GCC、Emacs 等&#xff09;&#xff0c;是 Linux 下常用的程序调试器。发展至今&#xff0c;GDB 已经迭代了诸多个版本&#xff0c;当下的…

C#上位机序列9: 批量读写+事件广播+数据类型处理

一、源码结构&#xff1a; 二、运行效果&#xff1a; 三、源码解析 1. 读取配置文件及创建变量信息&#xff08;点位名称&#xff0c;地址&#xff0c;数据类型&#xff08;bool/short/int/float/long/double&#xff09;&#xff09; 2. 异步任务处理&#xff1a;读任务&…

c++_learning-并发与多线程

并发与多线程 并发&#xff1a;进程&#xff1a;线程&#xff1a;基本概念&#xff1a;线程安全&#xff1a;问题出现的场景&#xff1a;涉及的性质&#xff1a;如何保证线程安全&#xff1f; 并发的实现手段&#xff08;优先使用多线程并发&#xff09;&#xff1a;多进程并发…

【特征重要性揭秘:为什么同一个数据集会有不同结果】

文章目录 特征重要性概要为什么特征重要性分析很重要特征重要性分析方法内置特征重要性(coef_或feature_importances_)Leave-one-out相关性分析递归特征消除 Recursive Feature EliminationXGBoost特性重要性主成分分析 PCA方差分析 ANOVA卡方检验&#xff08;Chi-Square Test&…

求最大公约数的几种常见的方法 【详解】

目录 一、关于公约数 二、计算最大公约数的方法 1. 辗转相除法&#xff08;欧几里得算法&#xff09; 2. 更相减损法&#xff08;辗转相减法&#xff09; 3. 分解质因数法 4. 穷举法 5. 递归法 6. 短除法 三、总结 一、关于公约数 首先 &#xff0c;先介绍一下公约…

matplotlib python 画图教程(2)

1、bar 柱状图 import matplotlib.pyplot as plt import numpy as np n12 xnp.arange(12) y1(1-x/float(n))*np.random.uniform(0.5,1,n) y2(1-x/float(n))*np.random.uniform(0.5,1,n) plt.xlim(-.5,n) plt.ylim(-1.25,1.25) plt.xticks([]) plt.yticks([]) plt.bar(x,y1,fac…

利用TypeScript 和 jsdom 库实现自动化抓取数据

以下是一个使用 TypeScript 和 jsdom 库的下载器程序&#xff0c;用于下载zhihu的内容。此程序使用了 duoip.cn/get_proxy 这段代码。 import { JSDOM } from jsdom; import { getProxy } from https://www.duoip.cn/get_proxy;const zhihuUrl https://www.zhihu.com;(async (…

01、Python 安装 ,Pycharm 安装

目录 安装安装 Python安装 Pycharm 创建项目简单添加文件运行 简单爬取下载小视频 安装 python-3.8.10-amd64.exe – 先安装这个 pycharm-community-2022.2.exe 再安装这个 安装 Python python-3.8.10-amd64.exe 安装&#xff08;这个是其他版本的安装&#xff0c;步骤一样…

Linux安装MINIO

MINIO简介MINIO目录 mkdir -p /opt/minio/data && cd /opt/minio MINIO下载 wget https://dl.minio.org.cn/server/minio/release/linux-amd64/minio MINIO授权 chmod x minio MINIO端口 firewall-cmd --zonepublic --add-port7171/tcp --permanent && firewal…

RTOS(6)任务管理

任务状态理论 我们是怎么实现&#xff0c;两个同优先级的任务之间交替执行的呢&#xff1f; 任务切换的基础&#xff1a;tick中断&#xff01; tick为1ms一个周期&#xff0c;可以通过修改时钟配置修改&#xff1b; running&#xff1a;正在进行的任务3为running&#xff…

20231019 filezilla 配置 Windows与Ubuntu文件传输

SFTP协议&#xff0c;传文件&#xff0c;否则会报无权限错

RTOS(7)同步互斥与通信概述

同步与互斥 同步的例子 循环检测有缺陷&#xff0c;还是得blocked掉&#xff0c;不然会很占用cpu&#xff0c;浪费资源&#xff1b; 互斥的例子 单纯的使用全局变量来实现互斥不太靠谱&#xff0c;当执行时间过长的时候会概率性的出现错误 通信的例子 FreeRtos的解决方案 …

Linux操作系统从BIOS到bootloader是如何运行的

操作系统一般都会在安装在硬盘上&#xff0c;在 BIOS 的界面上。你会看到一个启动盘的选项。启动盘有什么特点呢&#xff1f;它一般在第一个扇区&#xff0c;占 512 字节&#xff0c;而且以 0xAA55 结束。这是一个约定&#xff0c;当满足这个条件的时候&#xff0c;就说明这是一…

浅谈轨道交通建筑能耗分析及节能措施

叶根胜 安科瑞电气股份有限公司 上海嘉定 201801 摘要&#xff1a;面对城市轨道交通的能耗增长&#xff0c;优化地铁车站建筑、降低运营能耗是促进公共交通可持续化发展的必经之路。通风空调系统的能耗占比较大&#xff0c;节能潜力也是大的。本文以上海首条绿色地铁的项目实…

C#计数排序算法

前言 计数排序是一种非比较性的排序算法&#xff0c;适用于排序一定范围内的整数。它的基本思想是通过统计每个元素的出现次数&#xff0c;然后根据元素的大小依次输出排序结果。 实现原理 首先找出待排序数组中的最大值max和最小值min。创建一个长度为max-min1的数组count&a…

售后服务管理升级要怎么做?如何用好售后工单管理系统?

随着数字经济的发展&#xff0c;市场服务趋于多样化&#xff0c;客户对售后服务也有更高的要求&#xff0c;市场竞争日益激烈&#xff0c;越来越多的供应商企业认识到单凭优异的质量&#xff0c;颇具竞争力的价格已不能保证企业的竞争优势&#xff0c;事实表明&#xff0c;用户…

[原创] ElasticSearch集群故障案例分析: 警惕通配符查询

[携程旅行网&#xff1a; 吴晓刚] 许多有RDBMS/SQL背景的开发者&#xff0c;在初次踏入ElasticSearch世界的时候&#xff0c;很容易就想到使用(Wildcard Query)来实现模糊查询&#xff08;比如用户输入补全)&#xff0c;因为这是和SQL里like操作最相似的查询方式&#xff0c;用…