java的volatile

在Java中,线程之间对内存写入操作的可见性是一个重要的问题,因为每个线程都有自己的工作内存,并且线程之间共享主内存。当一个线程修改了共享变量的值,其他线程并不一定能立即看到这个修改,这就是所谓的可见性问题。

例如下面的例子:

线程t1修改共享变量flag的值并不能被线程t2获取到,线程t2一直没有终止。

为了解决这个问题,Java提供了volatile关键字和synchronized关键字来确保线程之间对内存写入操作的可见性。

volatile关键字

volatile关键字用于修饰变量 ,它确保了对volatile变量的写操作会立即被其他线程看到。当一个线程修改了一个volatile变量的值,新值会立即被写入主内存,并且其他线程会立即看到这个变化。这是因为volatile变量会禁止指令重排序 ,并且当一个线程读取一个volatile变量时,它会从主内存中读取,而不是从自己的工作内存中读取(当写一个volatile变量时,JMM会把该线程对应的本地内存中的共享变量值立即刷新回主内存中。当读一个volatile变量时,JMM会把该线程对应的本地内存设置为无效,重新回到主内存中读取最新共享变量)。

但是,volatile并不能 保证复合操作的原子性 ,例如count++这样的操作实际上包含三个步骤:读取、修改和写入。在多线程环境下,这些步骤可能会被其他线程打断,导致数据不一致。
synchronized关键字

synchronized关键字用于修饰方法或代码块 ,它确保同一时刻只有一个线程可以执行某个代码块或方法,从而实现了线程之间的互斥。当一个线程进入synchronized代码块或方法时,它会获取一个锁,其他尝试进入该代码块或方法的线程会被阻塞,直到锁被释放。这就保证了在同一时刻只有一个线程可以修改共享变量,从而避免了可见性问题。

此外,synchronized还确保了内存可见性,即当一个线程释放锁时,它会将修改后的共享变量的值刷新到主内存中,使得其他线程在获取锁并读取共享变量时,能够看到最新的值。

上述例子修改后:

相关推荐
皮皮林5514 小时前
Java性能调优黑科技!1行代码实现毫秒级耗时追踪,效率飙升300%!
java
冰_河5 小时前
QPS从300到3100:我靠一行代码让接口性能暴涨10倍,系统性能原地起飞!!
java·后端·性能优化
桦说编程7 小时前
从 ForkJoinPool 的 Compensate 看并发框架的线程补偿思想
java·后端·源码阅读
躺平大鹅9 小时前
Java面向对象入门(类与对象,新手秒懂)
java
初次攀爬者10 小时前
RocketMQ在Spring Boot上的基础使用
java·spring boot·rocketmq
花花无缺10 小时前
搞懂@Autowired 与@Resuorce
java·spring boot·后端
Derek_Smart11 小时前
从一次 OOM 事故说起:打造生产级的 JVM 健康检查组件
java·jvm·spring boot
NE_STOP12 小时前
MyBatis-mybatis入门与增删改查
java
孟陬16 小时前
国外技术周刊 #1:Paul Graham 重新分享最受欢迎的文章《创作者的品味》、本周被划线最多 YouTube《如何在 19 分钟内学会 AI》、为何我不
java·前端·后端