Java多线程 结果错误:a++会让加的次数减少

a++会让加的次数减少

如下的代码演示了, 原本期望是20000的结果, 却由于线程不安全, 导致了结果不符合预期.

创建了两个线程, 都去调用run方法的共享变量的i++的操作.
并且使用了join让主线程等待子线程.
主线程等待子线程执行完毕之后, 再打印执行的结果.


/**
 * 类名称:MultiThreadsError
 * 类描述:  第一种:运行结果出错。 演示计数不准确(减少),找出具体出错的位置。
 *
 * @author: https://javaweixin6.blog.csdn.net/
 * 创建时间:2020/8/31 19:18
 * Version 1.0
 */
public class MultiThreadsError implements Runnable{

    static MultiThreadsError instance = new MultiThreadsError();
    int index = 0;

    public static void main(String[] args) throws InterruptedException {
        Thread thread1 = new Thread(instance);
        Thread thread2 = new Thread(instance);
        thread1.start();
        thread2.start();

        //主线程 等待子线程
        thread1.join();
        thread2.join();

        //打印结果
        System.out.println(instance.index);
    }

    @Override
    public void run() {
        for (int i = 0; i < 10000; i++) {
            index++;
        }
    }

}

运行两次程序 可以看到分别打印如下. 结果是不一致的, 这就出现了线程不安全的现象.

加的次数减少原因分析

主要的原因是i++的操作, 看似是一条语句, 但其实是3条语句,
先要获取i的值, 再加1 ,最后再把加好的值,赋值给i.
那么就可能出现, 在线程1 执行了加1操作的时候, 还没把最新的值赋值给i , cpu调度就把执行权给了线程2 , 导致原本此时应该是i是2的,但是却获取到的是1的值. 这样就导致了加的值变小了.
如下图的的箭头执行流程,

猜你喜欢

转载自blog.csdn.net/qq_33229669/article/details/108327742