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

线程安全问题的原因和解决方案

原因

1.根本原因:线程的调度和执行是随机的(抢占式执行)

2.多个线程同时修改同一个变量

        ·多线程读取都没事

        ·多线程修改不同变量也没事

3.修改操作不是原子的

        补充:原子

        ·是不可拆分的最小单位

        ·在cpu执行指令的角度,执行一条指令这个做法就是原子的如果是一条指令,对于cpu来说要么就是执行完要么就是不执行,不会执行一半。

        ·=这个做法也是原子的。

        ·如果是count++这种对应了多条指令,就有可能cpu在执行过程中就执行一般,就调度走执行别人的指令,这样就不是原子的。

4.内存可见性

5.指令重排序

(主要还是前三条)

解决方案

·把修改操作变成原子的---加锁---打包成整体来达到整体性。

加锁:

synchronized (locker) {
//将操作放进锁里for (int i = 0; i < 10000; i++) {count++;}}

1.括号里放的是锁对象,要记得用之前先定义 

public static  Object locker=new Object();

2.object是锁对象的类型,取决于锁里面的东西是什么类型 

        eg:count++操作,count是int类型,要用object类型(或者用integer)

3.在加锁中,竞争同一把锁的时候才会产生“阻塞”---就看锁对象是不是同一个。

·如果是两个线程一个加锁一个没加那也不会阻塞

·两个线程都锁了而且是同一个对象才会产生锁竞争

import javax.swing.plaf.BorderUIResource;
//实现10000+10000正确显示20000
//注意解除优化volatile还有对里面的操作加锁
public class Demo1 {public static volatile int count=0;public static  Object locker=new Object();public static void main(String[] args) throws InterruptedException {Thread thread1=new Thread(()->{for (int i = 0; i < 10000; i++) {synchronized (locker) {count++;}}});Thread thread2=new Thread(()->{for (int i = 0; i < 10000; i++) {synchronized (locker) {count++;}}});thread1.start();thread2.start();thread1.join();thread2.join();System.out.println(count);}
}

·在10000+10000=20000的例子里面加锁就是要用同一个对象locker对内容count加锁,而且是二者的count都要加锁才行

        ·在++的过程里包含load-add-save-unlock

        ·t1加锁后t2的加锁不成功,阻塞等待,直到t1执行unlock释放锁

        ·t1加锁后:lock-load-add-save-unlock 从而t2执行的load到的数据就是t1已经save过的

import javax.swing.plaf.BorderUIResource;
//实现10000+10000正确显示20000
//注意解除优化volatile还有对里面的操作加锁
public class Demo1 {public static volatile int count=0;public static  Object locker=new Object();public static void main(String[] args) throws InterruptedException {Thread thread1=new Thread(()->{synchronized (locker) {for (int i = 0; i < 10000; i++) {count++;}}});Thread thread2=new Thread(()->{synchronized (locker) {for (int i = 0; i < 10000; i++) {count++;}}});thread1.start();thread2.start();thread1.join();thread2.join();System.out.println(count);}
}

        ·而对count加锁的方法除了直接锁count++之外还可以在for循环外加(这种方法是两个想你换串行的)t1不停执行循环,直到执行完10000次

        ·这两种方法中第一种比较好,便于充分利用cpu多核心资源;t1和t2谁拿到锁是不确定的。

        ·第二种没有把多核心利用起来

http://www.xdnf.cn/news/9727.html

相关文章:

  • String类中的常用方法
  • RapidOCR集成PP-OCRv5_det mobile模型记录
  • 【AI论文】ScienceBoard:评估现实科学工作流程中的多模态自主代理
  • 【FPGA开发】Ubuntu16.04环境下配置Vivado2018.3—附软件包
  • mysql执行sql语句报错事务锁住
  • Python爬虫实战:研究Aiohttp库相关技术
  • 【C语言】指针详解(接)
  • 游戏盾在非游戏行业的应用实践与价值分析
  • 立志成为一名优秀测试开发工程师(第九天)——使用fiddler工具、request库进行接口测试
  • GitCode镜像门法律分析:PL协议在中国的司法实践
  • Python 生成器:从基础到高级
  • 【Ubuntu】Ubuntu网络管理
  • Vscode 解决 #include <> 找不到的问题
  • x86_64-apple-ios-simulator 错误
  • 政策+技术双轮驱动:MiC建筑如何成为“好房子”建设的破局之道
  • UE5.5 pixelstreaming插件打包报错
  • UE5打包项目设置Project Settings(打包widows exe安装包)
  • 《深入解析UART协议及其硬件实现》-- 第三篇:UART ASIC实现优化与低功耗设计
  • 图像数据与显存
  • WebFuture:后台修改内容链接地址保存提示内容链接地址禁止输入外部url链接
  • Spring Boot自动装配原理
  • QT+opecv如何更改图片的拍摄路径
  • Oracle 慢sql排查
  • 前端面试准备2
  • Axure设计案例——科技感渐变柱状图
  • 24点游戏的最小数字组合问题
  • 常见关系型数据库对比指南
  • 制造业的未来图景:超自动化与劳动力转型的双重革命
  • 数据库中常见的锁机制详解
  • Java设计模式之迭代器模式详解