理解线程安全:保护你的代码免受并发问题困扰

news2025/1/16 17:30:17

目录

前言

一、什么是线程安全?

二、为什么需要线程安全?

三、实现线程安全的方法

四、synchronized

使用 synchronized 关键字时,需要注意以下几点:

五、Demo讲解


前言

        在现代软件开发中,尤其是在多线程编程中,线程安全(Thread Safety)是一个至关重要但又复杂的话题。本文将从基础概念开始,逐步深入,帮助你理解什么是线程安全,以及如何在实际开发中实现线程安全。

一、什么是线程安全?

        线程安全是指多个线程可以同时访问和修改共享数据而不导致数据的不一致性或程序错误。简单来说,当一个对象或函数在多线程环境下能够正确地执行,并且不会引起任何未定义的行为时,它就是线程安全的。

二、为什么需要线程安全?

        多线程编程的主要目的是为了提高程序性能和响应速度。然而,当多个线程同时操作共享资源时,可能会出现竞争条件(Race Condition)、死锁(Deadlock)以及饥饿(Starvation)等问题。这些问题不仅会导致程序运行结果不正确,还可能引发崩溃和难以调试的错误。

三、实现线程安全的方法

  1. 互斥同步:使用锁(如 synchronized、ReentrantLock 等)来保护共享数据,在同一时刻只允许一个线程访问共享数据,其他线程需要等待锁释放后才能访问。

  2. 原子操作:使用原子操作来确保对共享数据的操作是不可分割的,不会被中断,常见的原子操作包括 atomic 包下的原子类,以及 volatile 关键字修饰的变量。

  3. 无锁并发编程:通过使用无锁的数据结构和算法来实现线程安全,例如 CAS(Compare and Swap)操作,乐观锁机制等。

  4. 线程封闭:将共享数据限制在单个线程内部,避免多个线程之间直接访问共享数据,从而避免竞态条件。

  5. 不可变对象:通过创建不可变对象来避免多线程并发修改共享状态,从而避免线程安全问题。

四、synchronized

        当一个线程访问一个被 synchronized 修饰的方法或代码块时,会自动获取该方法或代码块对应的锁,其他线程必须等待该线程释放锁后才能获取锁并进入临界区。这种方式可以有效避免多个线程同时访问共享数据时发生竞态条件和其他线程安全问题。

使用 synchronized 关键字时,需要注意以下几点:

  1. 修饰方法:在修饰方法时,锁对象默认为当前对象(this),不同的线程需要获取的是同一个对象的锁。

  2. 修饰代码块:在修饰代码块时,需要指定锁对象,一般可以使用 this、类名.class 等作为锁对象。需要注意的是,锁对象必须是同一个对象才能保证线程同步。

  3. 可重入性:在同一个线程中,如果一个方法已经获取了锁,那么在调用另一个被 synchronized 修饰的方法时,该线程仍然能够获取到同一个锁,即 synchronized 具有可重入性。

  4. 锁的释放:在 synchronized 块执行完毕或者抛出异常时,锁会自动释放。如果在 synchronized 块中使用了 wait() 方法,则锁也会被释放,等待其他线程唤醒后再次获取锁。

五、Demo讲解

package com.ctb.demo;


/**
 * 线程安全概念:当多个线程访问某一个类(对象或方法)时,这个对象始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的
 * synchronized:可以在任意对象或方法上加锁,而加锁的这段代码称为"互斥区"或"临界区"
 * @author 彪
 *
 */
public class MyThread extends Thread{
	private int count = 5;
	
	
	public void run() {
		count--;
		System.out.println(this.currentThread().getName()+"count = "+count);
	}
	
	public static void main(String[] args) {
		/**
		* 	当多个线程访问MyThread的run方法时,以排队的方式进行处理(这里排队是按照CPU分配的先后顺序而定的)
		*   一个线程想要执行synchronized修饰的方法里的代码
		*   1.尝试获得锁
		*   2.如果拿到锁,执行synchronized代码内容:拿不到锁,这个线程就会不断尝试获得这把锁,直道拿到为止,
		*   	而且是多个线程同时去竞争这把锁。(也就是会有锁竞争的问题)
		*/
		MyThread myThread = new MyThread();
		Thread t1 = new Thread(myThread,"t1");
		Thread t2 = new Thread(myThread,"t2");
		Thread t3 = new Thread(myThread,"t3");
		Thread t4 = new Thread(myThread,"t4");
		Thread t5 = new Thread(myThread,"t5");
		t1.start();
		t2.start();
		t3.start();
		t4.start();
		t5.start();
	}

}

结果:并不是我们想要的,count--

package com.ctb.demo;


/**
 * 线程安全概念:当多个线程访问某一个类(对象或方法)时,这个对象始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的
 * synchronized:可以在任意对象或方法上加锁,而加锁的这段代码称为"互斥区"或"临界区"
 * @author 彪
 *
 */
public class MyThread extends Thread{
	private int count = 5;
	
	
	//synchronized加锁
	public synchronized void run() {
		count--;
		System.out.println(this.currentThread().getName()+"count = "+count);
	}
	
	public static void main(String[] args) {
		/**
		* 	当多个线程访问MyThread的run方法时,以排队的方式进行处理(这里排队是按照CPU分配的先后顺序而定的)
		*   一个线程想要执行synchronized修饰的方法里的代码
		*   1.尝试获得锁
		*   2.如果拿到锁,执行synchronized代码内容:拿不到锁,这个线程就会不断尝试获得这把锁,直道拿到为止,
		*   	而且是多个线程同时去竞争这把锁。(也就是会有锁竞争的问题)
		*/
		MyThread myThread = new MyThread();
		Thread t1 = new Thread(myThread,"t1");
		Thread t2 = new Thread(myThread,"t2");
		Thread t3 = new Thread(myThread,"t3");
		Thread t4 = new Thread(myThread,"t4");
		Thread t5 = new Thread(myThread,"t5");
		t1.start();
		t2.start();
		t3.start();
		t4.start();
		t5.start();
	}

}

结果:

注意:

当多个线程访问MyThread的run方法时,以排队的方式进行处理(这里排队是按照CPU分配的先后顺序而定的) ​ 一个线程想要执行synchronized修饰的方法里的代码 ​ 1.尝试获得锁 ​ 2.如果拿到锁,执行synchronized代码内容:拿不到锁,这个线程就会不断尝试获得这把锁,直道拿到为止, ​ 而且是多个线程同时去竞争这把锁。(也就是会有锁竞争的问题:前面线程count不是按顺序的)

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

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

相关文章

【源码】二开版微盘交易系统/贵金属交易平台/微交易系统

二开版微盘交易系统/贵金属交易平台/微交易系统 一套二开前端UI得贵金属微交易系统,前端产品后台可任意更换 此系统框架不是以往的至尊的框架,系统完美运行,K线采用nodejs方式运行 K线结算都正常,附带教程 资源来源:https://www.…

C++ UML建模

starUML UML图转C代码 数据流图 E-R图 流程图 整体架构图 ORM关系图 参考 app.asar附件资源可免激活 JHBlog/设计模式/设计模式/1、StarUML使用简明教程.md at master SunshineBrother/JHBlog GitHub GitHub - dimon4ezzz/whitestaruml: UML modeling tool derived from …

汇编语言期末复习

目录 前言 基础知识 80x86计算机组织 80x86的寻址方式 前言 根据老师的PPT与IBM-PC汇编语言程序设计(第2版)而写,供考前突击所用。 基础知识 q 机器语言、汇编语言、高级程序语言 特性 比较 q 进位记数制与不同基数的数之间的转换 二进…

可变参数以及不可变集合

可变参数: 格式: public class ArgsDemo {public static void main(String[] args) {System.out.println(getSum(1,2,3,4,5));}//可变参数public static int getSum(int...args){int sum 0;for (int arg : args) {sum arg;}return sum;} }可变参数的…

笨蛋学算法之LeetCodeHot100_1_两数之和(Java)

package com.lsy.leetcodehot100;public class _Hot1_两数之和 {//自写方法public static int[] twoSum1(int[] nums, int target) {//定义存放返回变量的数组int[] arr new int[2];//遍历整个数组for (int i 0; i < nums.length; i) {//从第二个数开始相加判断for (int j…

RK3588 Debian11进行源码编译安装Pyqt5

RK3588 Debian11进行源码编译安装Pyqt5 参考链接 https://blog.csdn.net/qq_38184409/article/details/137047584?ops_request_misc%257B%2522request%255Fid%2522%253A%2522171808774816800222841743%2522%252C%2522scm%2522%253A%252220140713.130102334…%2522%257D&…

SpringBoot内置数据源

回顾: 在我们之前学习在配置文件当中配置对应的数据源的时候, 我们设置的数据源其实都是Druid的数据源, 并且其配置有两种方式, 当然这两种方式都需要我们导入对应的有关 德鲁伊 的依赖才行 一种是直接在开始设置为 druid 数据源类型的一种是在对应的正常的数据库配置下, 设置…

51 USART数据收发

1.0 USART实现单个数据收发 串口启动之前需要对串口进行初始化&#xff0c;主要是设置产生波特率的定时器1&#xff0c;使用串口的工作方式还是中断的工作方式具体的配置步骤如下所示。 注&#xff1a; 1&#xff1a; 确定TMOD &#xff08;定时器模式寄存器&#xff09; 确…

Thinkpad产品系列进BIOS设置(重装系统)

Thinkpad产品系列进BIOS设置&#xff08;重装系统&#xff09; 对于大多数ThinkPad笔记本产品&#xff08;T、X、W、P、L、E系列部分除外&#xff09;&#xff0c;例如T14、T15、T490、T590、X13、X390等&#xff0c;您需要在启动计算机时&#xff0c;当显示ThinkPad徽标时&…

【简单讲解Perl语言】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…

46【Aseprite 作图】发光

1 通过“编辑 - 特效 - 卷积矩阵”&#xff0c;这次选择“7*7”&#xff0c;可以做出窗户的效果

【数据结构初阶】 --- 单链表

关于链表你应该先了解这些 下图描述了物理模型和逻辑模型&#xff0c;大多数常见的其实是逻辑模型&#xff0c;但这对初学者或者掌握不扎实的同学不太友好&#xff0c;所以这里我重点讲解物理模型&#xff0c;当了解了这些细节&#xff0c;以后做题或是什么就直接画逻辑模型就…

第8章 函数

第8章 函数 8.1 定义函数8.1.1 向函数传递信息8.1.2 实参和形参 8.2 传递实参8.2.1 位置实参8.2.2 关键字实参8.2.3 默认值 8.3 返回值8.3.1 返回简单值8.3.2 让实参变成可选的8.3.3 返回字典8.3.4 结合使用函数和 while 循环 8.4 传递列表8.4.1 在函数中修改列表8.4.2 禁止函数…

vue操作蓝牙教程

项目背景 想在VUE中使用蓝牙功能&#xff0c;百度了好久也尝试了好多都没法实现。 概念讲价 如果要在浏览器中使用蓝牙&#xff0c;去搜索关键字【navigator.bluetooth】&#xff0c;搜索后发现这根本不是想要的结果。 解决方法 去搜索关键字【uniappbluetoothvue】&#x…

mouceMice智能垃圾分类系统

mouceMice智能垃圾分类系统 1.成员名称和任务分配 成员认领任务陈曦实现登录、注册、检索垃圾类型和前端部分实现、连接树莓派实现socket通信传输图片杨雨佳需求分析和总体进展监督郑博文部分前端页面实现及其优化李睿初步实现深度学习算法、树莓派连接和算法效率提升范兴宇项…

【AI法官】人工智能判官在线判案?

概述 AI法官是一款为用户提供专业法律分析和判决建议的智能体应用。用户只需简要描述案情&#xff0c;AI法官便会利用其强大的法律知识和逻辑推理能力&#xff0c;快速且准确地梳理出判决结果。该应用的目标是为用户提供高效、准确、合法的判决建议。 角色任务 任务描述 作为…

【CS.SE】2024年,你应该选择计算机专业吗?详细分析与未来展望

文章目录 1. 引言1.1 背景介绍 2. 计算机相关专业的现状与挑战2. 计算机相关专业的现状与挑战2.1 行业内的就业趋势2.1.1 现有就业数据2.1.2 行业需求变化 2.2 市场饱和度与竞争2.2.1 毕业生数量增长2.2.2 薪资与职业发展 2.3 技术创新与行业发展2.3.1 新兴技术的发展2.3.2 全球…

TinyHttpd源码精读(三)

在上一章中我们一起看了如何实现静态的网页&#xff0c;在这里我们一起看Tinyhttpd最后的一部分&#xff0c;动态网页的实现&#xff1a;在这里首先声明下因为cgi脚本的支持问题&#xff0c;所以我会新建一个简单的cgi脚本然后将路径导向到这个脚本&#xff1a; 0.perl的配置&…

2024年建筑、水利交通与工程管理国际学术会议(ICAWRTEM 2024)

全称&#xff1a;2024年建筑、水利交通与工程管理国际学术会议&#xff08;ICAWRTEM 2024&#xff09; 会议网址:http://www.icawrtem.com会议地点: 广州投稿邮箱&#xff1a;icawrtemsub-conf.com 投稿标题&#xff1a;ICAWRTEM 2024ArticleTEL。投稿时请在邮件正文备注&#…

MySQL之高级特性(一)

高级特性 外键约束 InnoDB是目前MySQL中唯一支持外键的内置存储引擎&#xff0c;所以如果需要外键支持那选择就不多了。使用外键是有成本的。比如外键通常都要求每次在修改数据时都要在另一张表中多执行一次查找操作。虽然InnoDB强制外键使用索引&#xff0c;但还是无法消除这…