小编典典

整数对象的同步块

java

我刚刚遇到了Java中的同步块,并编写了一个小程序来测试它的工作方式。

我创建10个线程,并让每个线程将一个Integer对象增加1000次。

因此,在使用同步的情况下,所有线程完成工作后,我将假定结果为10000,而在没有同步的情况下,结果将小于10000。

但是,同步并没有如我所料。

我猜想这与对象的不变性有关。

我的程序:

public class SyncTest extends Thread{

    private static Integer syncObj = new Integer(0);
    private static SyncTest[] threads = new SyncTest[10];

    private boolean done = false;

    public void run(){
        for(int i = 0; i < 1000; i++){
            synchronized(syncObj){
                syncObj ++;
            }
        }
        done = true;
    }

    public static void main(String[] args) {

        for(int i=0; i < threads.length; i++){
            threads[i] = new SyncTest();
            threads[i].start();
        }

        while(!allDone()); //wait until all threads finished

        System.out.println(syncObj);
    }

    private static boolean allDone(){
        boolean done = true;
        for(int i = 0; i < threads.length; i++){
            done &= threads[i].done; 
        }

        return done;
    }
}

有人可以澄清吗?


阅读 211

收藏
2020-09-18

共1个答案

小编典典

每次您对进行同步时,syncObject都会发生变化(会将其转换为原始int,对其进行递增,然后将其自动装箱回到Integer对象。Integer对象是不可变的……一旦创建,它们就无法更改。

最糟糕的是,您并没有在所有线程中使用相同的syncPObj,不同的线程在不同的时间使用不同的syncObject进行同步。

使用一个对象作为同步(称为syncObj),并将其声明为最终对象:

private static final Object syncObject = new Object();

然后,您的计数器应该是用于性能的原始(int),称为“计数器”之类的东西。

在syncObject上同步,并增加计数器。

编辑:根据@jsn,完成标志也被破坏,因为您的代码在isAllDone()方法上有一个“紧循环”,这是一种不好的做法。您应该使用thread [i]
.join()等待(阻塞)每个线程的完成,然后从中检查状态。使用ExecutorService是“正确的方法”。

2020-09-18