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

并发编程出现的问题以及解决方式

解决并发编程出现的问题

基于java内存模式的设计出现的问题

基于java内存模式的设计,多线程操作一些共享的数据时,出现以下三个问题:

1.不可见性问题:多个线程同时在各自的工作内存对共享数据进行操作,彼此之间不可见。操作完写会主内存,有可能出现问题。

2.无序性:为了性能,对一些代码指令的执行顺序调整重排,以提高速度。在某种情况下,顺序调整后,可能会对后续代码操作进行影响。

3.非原子性:对进程的程序代码分割进行了,由于线程切换而导致

缓存(工作内存)带来了不可见性;
指令重排优化,带来了无序性;
线程切换,带来了非原子性;

解决办法

让不可见变为 可见:各自的工作内存的共享数据可以实时刷新

让无序变为 不乱序:不对代码重新排序

非原子执行 变为原子:加锁

1.实现可见性和有序性

volatile关键字

volatile修饰的是变量

解决了两个问题

volatile所修饰的变量被一个线程修改后,可以在其他线程中立即可见。可解决不可见问题

volatile修饰的变量,在执行的过程中与它相关的代码不会被重排序执行。可解决无序性问题

但volatile不能解决原子性问题

volatile 底层实现原理

在底层指令级别来进行控制

volatile修饰的变量在操作前,添加内存屏障,不让其他的指令干扰。

volatile修饰的内存变量添加内存屏障之外,还要通过缓存一致性协议(MESI)将数据写回到主内存,其他工作内存嗅探后把自己工作内存数据过期,重新从主内存读取最新的数据。

2.实现原子性

(1)加锁

通过加锁的方式,让程序互斥执行来保持一次只有一个线程对共享资源访问。

加锁的两种方式:

synchronized:关键字 修饰代码块,方法 自动获取锁、自动释放锁

Reentrantlock:类 只能修饰代码块 手动加锁、释放锁

(2)使用原子类(非加锁)

在java中,还提供了一些原子类,是一种无锁实现;在低并发情况下使用;采用了CAS机制(Compare-And-Swap)

原子类的原子性是通过 volatile + CAS 实现原子操作的。 如AtomicInteger类,AtomicInteger 类中的 value 是有 volatile 关键字修饰的,这就保证了 value的内存可见性,这为后续的 CAS 实现提供了基础。

这里说一下CAS机制:

CAS(面试中出现频率很高)

CAS机制(Compare-And-Swap),比较并交换,该算法是硬件对于并发操作的支持;

是乐观锁的一种实现方式;

特点:

  • 是一种无锁实现;

  • 只能在低并发情况下使用;

  • 不加锁,所有线程都可以对共享数据操作;

  • 由于不加锁,所以不会阻塞,效率比加锁高;

  • 采用自旋思想;

自旋思想:

第一次采取内存值到工作内存中,存储起来作为预期值。然后对象数据进行修改,将工作内存值写入到主内存;

在写入之前需要做一个判断,用预期值与主内存中的值进行比较,如果预期值与主内存中值一致,说明这个变量没有其他线程修改,则将更新后的值,写入到主内存;

如果预期值与主内存中值不一致,说明其他进行修改了主内存的值,这时就需要重复这个过程;

概念:即每次判断我的预期值A和内存中的值V是不是相同,如果不相同则说明该内存值已经被其他线程更新过了,因此需要拿到该最新值B作为预期值,重新判断。而该线程不断的循环判断是否该内存值已经被其他线程更新过了。

CAS缺点

CAS使用自旋锁的方式,由于该锁会不断循环判断,因此不会synchronize线程阻塞导致线程切换,但是会不断自旋,导致cpu的消耗,在并发量大的时候导致cpu跑满。

导致ABA问题,通过设置版本号,每次操作改变版本号即可

http://www.lryc.cn/news/3379.html

相关文章:

  • [ linux ] linux 命令英文全称及解释
  • C++11新特性
  • 【宝塔部署SpringBoot前后端不分离项目】含域名访问部署、数据库、反向代理、Nginx等配置
  • 从0到1一步一步玩转openEuler--11 openEuler基础配置-设置磁盘调度算法
  • 河道治理漂浮物识别监测系统 yolov7
  • 微信小程序 java ssm Springboot学生作业提交管理系统
  • 实战项目-课程潜在会员用户预测(朴素贝叶斯&神经网络)
  • ESP32设备驱动-定时器与定时器中断
  • 【JavaScript 逆向】安居客滑块逆向分析
  • 【STM32】【HAL库】遥控关灯1主机
  • Java 初始化块
  • 超详细讲解长度受限制的字符串函数(保姆级教程!!!)
  • 【c#】c#常用小技巧方法整理(4)——cmd命令提示符,c#调用cmd
  • 在项目中遇到的关于form表单的问题
  • 德国奔驰、博世和保时捷的员工年薪有多少?
  • Mybatis与微服务注册
  • JAVA练习47-合并两个有序数组
  • 右键菜单管理 - Win系统
  • 背包问题求方案数、具体方案
  • 电商导购CPS,淘宝联盟如何跟单实现用户和订单绑定
  • 【Shell1】shell语法,ssh/build/scp/upgrade,环境变量,自动升级bmc,bmc_wtd,
  • 刷题记录:牛客NC208250牛牛的最美味和最不美味的零食
  • 微搭低代码从入门到精通08-轮播容器
  • 分类预测 | MATLAB实现SSA-CNN麻雀算法优化卷积神经网络多特征分类预测
  • 华为10年经验测试工程师,整理出来的python自动化测试实战
  • OpenCV杂谈 - 如何导出图像到内存中其他结构
  • Session与Cookie的区别(四)
  • Linux 文件锁 - fcntl
  • CellularAutomata元胞向量机-2-初等元胞自动机MATLAB代码分享
  • OpenStack云平台搭建(6) | 部署Neutron