Java中多线程安全问题实例分析

  • 2020 年 12 月 16 日
  • 笔记

案例


 1 package com.duyang.thread.basic.basethread;
 2 
 3 /**
 4  * @author :jiaolian
 5  * @date :Created in 2020-12-16 14:02
 6  * @description:线程不安全分析
 7  * @modified By:
 8  * 公众号:叫练
 9  */
10 public class ThreadUnsafe {
11 
12     public static void main(String[] args) {
13         Thread task = new Task();
14         Thread threadA = new Thread(task,"A");
15         Thread threadB = new Thread(task,"B");
16         Thread threadC = new Thread(task,"C");
17         Thread threadD = new Thread(task,"D");
18         Thread threadE = new Thread(task,"E");
19         threadA.start();
20         threadB.start();
21         threadC.start();
22         threadD.start();
23         threadE.start();
24     }
25 
26     private static class Task extends Thread {
27 
28         int count = 5;
29         @Override
30         public void run() {
31             /**
32              * jvm分3步骤;
33              * 1.获取count(从主内存获取值)
34              * 2.count减1(在各自寄存器完成)
35              * 3.保存count(刷新到主内存)
36              *
37              * 说下可能执行的过程...
38              * A线程获取cpu的count值为5,A线程先减去1,保存count值为4刷新到主内存,此时还没有执行System.out.println count
39              * 切换到B线程,此时B线程的count值为4,因为B线程是从主内存取的,B线程count值减去1为3,此时刷新到主内存,主内存值变为3
40              * 切换到A线程,执行System.out.println count=3
41              * 切换到B线程,执行System.out.println count=3
42              * 情况就是这样的
43              *
44              */
45             count--;
46             System.out.println(Thread.currentThread().getName() + " "+count);
47         }
48     }
49 }

 

 

可能的结果


结果得到下图(结论1图)

image.png

按理说应该是这样的啊

image.png

对,你想的没错,但是线程A,B的count值都等于3也是有可能的,下面我们来分析下。

 

详细分析


对于代码中45行,i–其实在JVM中,其实可以分为3步。

  • 获取count值(从主内存获取值)
  • count减1(在各自寄存器完成)
  • 保存count(刷新到主内存)

 

详细说下A,B实际上在机器中过程

  • A线程获取cpu的count值为5,A线程先减去1,保存count值为4刷新到主内存,此时还没有执行System.out.println打印count值。如下图所示

image.png

  • 切换到B线程,此时B线程的count值为4,因为B线程是从主内存取的,B线程count值减去1为3,此时刷新到主内存,主内存值变为3

image.png

  • 切换到A线程,执行System.out.println count=3
  • 切换到B线程,执行System.out.println count=3
  • C D E线程正常执行

 

这就是<结论1图>的执行过程。

 

结论


多线程安全一直是个很重要的话题,希望大家都能尽快理解掌握,希望大家喜欢!

我是叫练,多叫多练,欢迎大家和我一起讨论交流,我会尽快回复大家,喜欢点赞哦。