关于对volatile和wait()、notify方法的理解

一.情景引入:当我们运行以下场景:

当线程2修改num的值后,线程1仍然没有停止执行。

问题解决

想要解决这个问题,我们就必须要了解JMM(javaMemoryModel)java内存模型,JMM要求满足三个原则:①原子性②内存可见性③有序性,而这种问题的产生也是由于内存可见性:JMM规定如果一个线程想要操作主内存中的共享变量,必须将该变量load到其工作内存进行修改,修改之后再从其工作内存保存到主内存,这也就导致了问题,当线程1读取到原来主内存中的变量数据而线程2将其进行修改,但是线程1并不可见(当加载到工作内存时反复读的是之前缓存到工作内存中的数据),所以线程1无法停止。

volatile工作原理

那该如何解决该问题呢?只要能保证线程2中修改后的数据能被线程1接收,那么问题也就迎刃而解了,所以MESI缓存一致性协议随之诞生。简单来说,MESI可以被理解为一种通知机制,即当一个线程将主内存中的共享数据进行修改后,其他线程也能发现其修改,然后在主内存中调用最新的数据。而当一个变量被volatile修饰时cpu则会执行MESI缓存中的一系列操作
其工作原理是利用四种内存屏障,当一个线程中的共享数据进行修改时,其他线程中的工作内存被通知之前保存的数据失效(invalid),需要从主内存中重新读取数据(利用一个例子简单说明,详情请查阅该博客) https://blog.csdn.net/fuqianming/article/details/120307352 ,这样就能保证每个线程都是获取的最新的数据,也就保证了内存可见性
②.利用内存屏障禁止指令重排序(因为内存屏障为了保证数据的有效性事实上是在数据操作中添加了一些指令和限制,保证被volatile修饰的变量相关指令的执行顺序,如果进行了指令的重排序,那么这些限制也有可能被打乱)。
③不能保证原子性

wait()和notify()

wait ()和notify()方法的作用,在于人为影响线程的执行顺序

wait():当线程执行到这里时进行等待,让其他线程去准备相关的数据,当数据准备好后被唤醒,从wait()之后继续执行。

notify():准备相关的资源,准备好后,将wait()唤醒

使用事项:

wait()和notify()必须被synchronized包括,并且必须是同一个锁对象,否则会报错(非法的锁状态)

注意:

①当一个线程调用wait()方法后,其会释放锁,并等待其他线程唤醒

②当另一个线程调用notify()方法后,调用wait()的线程会被唤醒,并重新竞争锁资源,竞争到锁资源之后,从wait之后向下执行。

③notify和notifyall的区别:notify唤醒一个wait线程,notifyall唤醒所有的wait线程

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

六子干侧开

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值