多线程3(线程安全问题及解决方案)

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

2.解决方法


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

问题引入:当两个线程同时对变量count同时各自++50000次

正常来说,得到的结果应该为10_0000,但实际得到的结果有出入

因为++这个操作在操作系统执行的过程中分为三条指令:load,add,save

这三条指令他们不是原子性的,两个线程每次执行++操作,由于CPU的调度是随机的,指不定线程1还没执行完一个完整的++操作,CPU就调度了线程2,那么就会产生指令不连续,也就是被插队的情况,从而导致不能加到10_0000,那么这就叫线程安全问题

总结原因有以下几条:

(1)随机调度

(2)多线程同时修改同一成员变量

(3)修改操作不是原子的

(4)内存可见性(不讲)

(5)指令重排序(不讲)


2.解决方法

针对上述问题,我们目前只能对原因(3)下手

解决修改操作不是原子的问题

采用加锁,把修改的这个三条指令捆绑为一个完整的指令,不允许中途被人插队

synchronized(加锁)

使用synchronized时要对同一个同一个对象进行加锁

这样才能产生锁竞争:线程1获取到锁,线程2此刻也想获取锁就得阻塞等待,线程1实现了防止被线程2插队,直到线程1释放锁才能获取到锁

情况1:一个线程加锁另外一个线程不加锁 就相当于追妹子挖墙脚

情况2:加锁位置的不同,这样锁的粒度也会不同

什么叫锁的粒度:锁内的代码逻辑复杂度与锁粒度成正比

问题:为什么synchronized的作用范围是{},其他语言则采用lock()->unlock()的方式

Java的加锁确保了锁内的操作能完整的执行完毕且**释放锁,**而lock加unlock的方式弊端在于要采用unlock手动释放锁,那么就不靠谱,可能代码执行的过程逻辑出错,或者程序员忘记手动释放锁,那么就会出现问题:占着茅坑不拉shit

加锁是有开销的:加锁的代价就是效率降低,所以不得不加锁的时候才加锁,平时能不加锁就不加锁


相关推荐
代码羊羊21 小时前
Rust 迭代器完全通俗易懂指南(零基础全覆盖)
java·开发语言·rust
MY_TEUCK1 天前
【Java 后端】SpringBoot 登录认证与会话跟踪实战(JWT + Filter/Interceptor)
java·开发语言·spring boot
今天长肉了吗1 天前
银行风控项目踩坑实录:指标跑了6小时,风险评分全挂了
java
QQ2422199791 天前
基于python+微信小程序的家教管理系统_mh3j9
开发语言·python·微信小程序
随读手机1 天前
多式联运信息交互平台完整方案(2026版)
java·ai·eclipse·云计算·区块链
沐知全栈开发1 天前
JavaScript 条件语句
开发语言
RSTJ_16251 天前
PYTHON+AI LLM DAY THREETY-SEVEN
开发语言·人工智能·python
清水白石0081 天前
《Python性能深潜:从对象分配开销到“小对象风暴”的破解之道(含实战与最佳实践)》
开发语言·python
Je1lyfish1 天前
CMU15-445 (2025 Fall/2026 Spring) Project#3 - QueryExecution
linux·c语言·开发语言·数据结构·数据库·c++·算法
许彰午1 天前
03-二叉树——从递归遍历到非递归实现
java·算法