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

如何保证多线程安全

多线程安全的概念

关于安全的判定是指: 在多线程环境下执行的结果, 与在单线程下执行的结果保持一致即可认为是安全的

引发线程不安全的原因

最大的原因就是线程之间的执行顺序是由操作系统随机调度的,此时就会有以下几种情况导致线程不安全

1.可见性: 当涉及到多个线程对同一个数据修改时, 很有可能会导致因为一个线程修改后另外一个线程没拿到数据修改后的值的情况. 这是因为JMM的内存模型导致了这种情况.

2.原子性: 例如执行a++操作时, 是先拿到a的值,再将a+1,然后再将+1后的值赋值给a. 分为这三步. 但线程执行的顺序是随机的(A线程在执行a+1时有可能B线程突然出现执行了a-1). 解决这种问题需要将操作转化成一步就做完中间不能被其他线程插入.

3.有序性: 当线程需要执行的指令很多时, 系统会自动的对这些指令进行顺序上的优化.不会按照写的代码顺序进行执行. 而有时候这种优化会导致线程间的冲突.

解决线程不安全的方法

1.synchronized关键字

synchronized特性:

1.互斥性:进入或使用被synchronized修饰的方法和变量时,相当于是进行了 "'加锁操作". 此时其他线程无法进行操作

2.可重入性: 当一个线程进行了"加锁操作"后, 如果线程出现崩溃等情况时,锁得不到释放,那么其他线程就无法获取到锁了. 这就是"不可重入性".

synchronizd就是可重入锁. 在可重⼊锁的内部, 包含了 "线程持有者" 和 "计数器" 两个信息.
如果某个线程加锁的时候, 发现锁已经被⼈占⽤, 但是恰好占⽤的正是⾃⼰, 那么仍然可以继续到锁,
并让计数器⾃增. 每解一次锁的时候计数器减一. 直到递减为 0 的时候, 才真正释放锁. (才能被别的线程获取到)

synchronized的用法:

//对object对象进行加锁 public class SynchronizedDemo { private Object locker = new Object(); public void method() { synchronized (locker) { } } } //对当前对象进行加锁 public class SynchronizedDemo { public void method() { synchronized (this) { } } } //锁静态方法 public class SynchronizedDemo { public synchronized static void methond() { } }

2.volatile关键字

volatile解决的是"可见性"问题, 其中实现的原理是根据JMM的模型结构来进行操作的:

代码在写入volatile修饰的变量时:

改变工作内存的volatile变量副本的值, 将改变后的副本的值刷新到主内存中代码

在读取volatile修饰的变量时:

从主内存中获取volatile变量的最新值到工作内存中, 再从工作内存中拿到volatile变量的副本

3.使用JUC工具包(java.util.concurrent)

1. ReentrantLock(可重入锁)

ReentrantLock是Lock接口的实现类,功能和Synchronized类似,但支持公平锁 / 非公平锁可中断锁超时获取锁等高级特性。

import java.util.concurrent.locks.ReentrantLock; public class LockCounter { private int count = 0; // 创建可重入锁(默认非公平锁,公平锁传true:new ReentrantLock(true)) private final ReentrantLock lock = new ReentrantLock(); public void increment() { lock.lock(); // 获取锁(必须手动加锁) try { count++; // 核心业务逻辑 } finally { lock.unlock(); // 释放锁(必须放finally,避免异常导致锁泄漏) } } public int getCount() { lock.lock(); try { return count; } finally { lock.unlock(); } } }

ReentrantLock的功能比Synchronized更多, 但必需要自己进行加锁,释放锁.

2.使用Atomic类进行"原子性"操作

import java.util.concurrent.atomic.AtomicInteger; public class AtomicCounter { // 原子整数:替代int,保证自增/自减的原子性 private final AtomicInteger count = new AtomicInteger(0); public void increment() { count.incrementAndGet(); // 原子自增(等价于count++) } public int getCount() { return count.get(); // 原子读取 } }

使用Atomic类的CAS 算法(Compare-And-Swap)实现无锁同步,进行原子性操作性能极高, 适用于简单的数据更新场景

http://www.jsqmd.com/news/488895/

相关文章:

  • Carsim与Matlab/Simulink联合仿真在四轮电动汽车转向失效容错控制中的应用
  • 【笔记】用cursor手搓cursor(二)也就是龙虾育种
  • Fish Speech 1.5实战案例:游戏NPC多角色语音批量克隆工作流
  • 基于比例谐振型自抗扰控制GI ADRC的谐波抑制仿真模型
  • Qwen3-14b_int4_awq部署避坑:常见vLLM启动失败原因与Chainlit连接超时解决
  • 使用KNN - shap分类器解决多分类问题:基于Python自带数据集的实践
  • 搭建虚拟环境Linux
  • BootLoader上位机源码,HEX烧录刷写,基于LabVIEW和USBCAN FD-200U开发BootLoader刷写
  • uni-app Android应用在华为应用市场上架时隐私权限声明的最佳实践
  • 【小白量化机器人】部署本地大模型及设计一个【可爱聊天助手】
  • WeKnora问题解决:如何让AI严格按你给的文本回答问题
  • 嵌入式——04 网络编程
  • π型衰减器:高速ADC信号链的宽带匹配与无源衰减设计精要
  • FireRedASR Pro多语言识别效果评测:中英日韩等语种实测
  • 探索基于三菱 PLC 和组态王的锅炉控制系统
  • 探索SLM增材制造的ANSYS Fluent模拟之旅
  • 微信小程序横屏模式下登录强制竖屏后的界面方向恢复方案
  • 10分钟带你快速搭建第一个云服务器,手把手教程
  • Git LFS实战指南:高效管理Unity项目中的大型资源文件
  • 影墨·今颜实操案例:为国货美妆品牌定制‘水墨腮红’风格图
  • 嵌入式——05 C++
  • Phi-3-vision-128k-instruct模型微调实战:使用自定义数据提升特定场景识别能力
  • 比迪丽LoRA模型Java开发集成指南:SpringBoot后端服务调用
  • 5G NR 时频资源与帧结构:从理论到部署的深度解析
  • 从图片到Word:数学公式的高效转换技巧
  • 嵌入式——06 QT
  • 2026年评价高的芝麻黑品牌推荐:芝麻黑套碑/芝麻黑石材/芝麻黑大碑源头厂家推荐 - 品牌宣传支持者
  • 手把手教你用嘎嘎降AI处理毕业论文:从上传到下载全流程
  • Win10系统 通过注册表彻底禁用UAC实现全局管理员权限
  • cv_unet_image-matting镜像新手指南:从上传到下载完整流程