一个经典的并发导致异常的场景

Posted 梁桦

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了一个经典的并发导致异常的场景相关的知识,希望对你有一定的参考价值。

源码:

package lsh.concurrency.examples;

public class Novisibility {
    private static boolean ready;
    private static int number;
    private static int count = 0;
    
    private static class ReaderThread extends Thread{
        @Override
        public void run(){
            while(!ready){
                count++;
                Thread.yield();
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("count= "+count+"; ready= "+ready +" ;number= "+number);
            }
        }
    }
    
    public static void main(String[] args) {
            new ReaderThread().start();
            number = 42;
            try {
                System.out.println("开始休息前, ready= "+ready);
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            ready = true;
            System.out.println("休息够了, ready = "+ready );
    }

}

执行结果:

开始休息前, ready= false
count= 1; ready= false ;number= 42
count= 2; ready= false ;number= 42
count= 3; ready= false ;number= 42
count= 4; ready= false ;number= 42
休息够了, ready = true
count= 5; ready= true ;number= 42

特别留意最后一行里的ready,值为 true。while进入循环的条件明明是ready = false,但是最后一个输出结果怎么是true?

ps. 不是每次输出结果都会像上面那样。

 

 

原因分析:

while循环最后1秒的时候,主线程里的, ready = true; 这行代码先于子线程里while循环语句的输出。故主线程先修改了ready的值为true,等到子线程再次去读ready的值时(已经执行完了while的循环条件判断,判断时时ready的值为false),第二次读ready得到的值已经是true了。

 

 

补充:如果在while循环的输出之前,将ready = true时的值保存到一个文件里,会发现难易拿到ready的值,因为此时ready的值仍为false; 而如果在while循环的输出之后,对ready = true时,将其值保存到一个文件里,是一件相对容易的事情,因为此时ready值已经为true。

以上是关于一个经典的并发导致异常的场景的主要内容,如果未能解决你的问题,请参考以下文章

全栈编程系列SpringBoot整合Shiro(含KickoutSessionControlFilter并发在线人数控制以及不生效问题配置启动异常No SecurityManager...)(代码片段

几个关于js数组方法reduce的经典片段

片段中的getView()导致抛出异常,不确定原因

关闭从另一个对话框片段启动的对话框片段会导致非法状态异常

异常和TCP通讯

Fragment 和 FragmentStatePagerAdapter 中带有 ViewPager 的片段导致异常(带有完整示例)