• 可见性volatile


    Java内存模型

    JMM即Java Memory Model,它定义了主存(多个线程之间共享的内容)、工作内存(单个线程独自占有的内容)的抽象概念,底层对应着cpu寄存器、缓存、硬件内存cpu指令优化等。

    JMM体现在以下几个方面。

      * 原子性:保证指令不受到线程上下文切换的影响。

      * 可见性:保证指令不会受到cpu缓存的影响。

      * 有序性:保证指令不会受到cpu指令并行优化的影响。

    退不出的循环

    为什么呢?

    1.初始状态,t线程刚开始从主存读取了run的值到工作内存

    2.因为t线程要频繁从主存中读取run的值,JIT编译器会将run的值缓存至自己工作内存中的高速缓存中,减少对主存中run的访问,提高效率。

     3.1秒之后,main线程修改了run的值,并同步至主存,而t线程是从自己工作内存中的高速缓存中读取这个变量的值,结果永远是旧值。


    解决办法

    volatile(易变关键字:不可解决原子性问题)

    它可以用来修饰成员变量和静态成员变量(局部变量不存在多个变量共享的问题。)他可以避免线程从自己的工作缓存中查找变量的值,必须到主存中获取它的值,线程操作volatile变量都是直接操作主存。

    1. public class test {
    2. volatile static boolean run = true;
    3. public static void main(String[] args) throws InterruptedException {
    4. new Thread(() -> {
    5. while (run){
    6. // ...
    7. }
    8. }).start();
    9. Thread.sleep(1000);
    10. run = false;
    11. }
    12. }

    synchronized解决可见性问题

    1. public class test {
    2. static boolean run = true;
    3. //锁对象
    4. final static Object lock = new Object();
    5. public static void main(String[] args) throws InterruptedException {
    6. new Thread(() -> {
    7. while (true){
    8. synchronized (lock){
    9. if (!run){
    10. break;
    11. }
    12. }
    13. }
    14. }).start();
    15. Thread.sleep(1000);
    16. synchronized (lock){
    17. run = false;
    18. }
    19. }
    20. }

    synchronized和volatile都能保证共享变量的可见性的问题。但是在发生线程冲突的时候synchronized锁膨胀,变成重量级锁

    ReentratLock也可以解决可见性

    1. import java.util.concurrent.locks.ReentrantLock;
    2. public class test {
    3. static boolean run = true;
    4. static ReentrantLock reentrantLock = new ReentrantLock();
    5. public static void main(String[] args) throws InterruptedException {
    6. new Thread(() -> {
    7. while (true){
    8. reentrantLock.lock();
    9. try {
    10. if (!run){
    11. break;
    12. }
    13. }finally {
    14. reentrantLock.unlock();
    15. }
    16. }
    17. }).start();
    18. Thread.sleep(1000);
    19. reentrantLock.lock();
    20. try {
    21. run = false;
    22. }finally {
    23. reentrantLock.unlock();
    24. }
    25. }
    26. }


     

     

  • 相关阅读:
    仕兰微、华为、汉王、凹凸科技、南山之桥、周立功等公司常见FPGA面试题整理
    音视频 fmpeg命令裁剪和合并视频
    模块化设计瞎谈
    【元宇宙欧米说】数藏的社交与艺术收藏价值探讨
    世界粮食日:宏工科技有对策,赋能食品生产高效可持续发展
    【社媒营销】四大跨境营销渠道分析
    第七章:单链表与双链表(数组模拟)
    基于Qt mqtt库的客户端实现
    专业音视频领域中,Pro AV的崛起之路
    2020CCPC 威海站 个人题解
  • 原文地址:https://blog.csdn.net/weixin_54401017/article/details/126908559