多线程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

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


相关推荐
覆东流2 小时前
第2天:Python变量与数据类型
开发语言·后端·python
FeBaby2 小时前
Java 高并发场景下 Redis 分布式锁(UUID+Lua)最佳实践
java·redis·分布式
Gofarlic_oms12 小时前
制定企业Citrix虚拟化软件资产管理政策框架
运维·服务器·开发语言·matlab·负载均衡
落子君2 小时前
设计模式之【 断路器模式】
java
添砖java。。。2 小时前
java实现mqtt链接并控制门锁设备
java·开发语言
codeejun2 小时前
每日一Go-53、Go微服务--限流与降级
开发语言·微服务·golang
xier_ran2 小时前
【C++】static 关键字与 const 关键字的作用
java·数据库·microsoft
阿里嘎多学长2 小时前
2026-04-17 GitHub 热点项目精选
开发语言·程序员·github·代码托管
Wadli2 小时前
集群C++聊天服务器
服务器·开发语言·c++