`

主线程结束子线程也不执行了,很奇怪,按理不应该这样

 
阅读更多
引用
最近在测试使用多线程从数据库中读数据,所有子线程结束后,再讲汇总的数据保存到文件中。发现了一个问题,总是会出现这样一种情况:主线程结束后,子线程中的代码还没有执行结束就不再执行了,只执行了其中的一部分。

找了各种资料,都在说主线程结束后,如果子线程不是守护线程,子线程将不会受到影响,继续执行。我这里的线程明明不是守护线程,所以还是很奇怪。

最终终于找出原因了,原来是因为junit单元测试的原因,如果把这段代码放到main方法中执行,每次都是好的。

我还学到了CyclicBarrier这个类的使用方法和场景,这个类在多线程时还是非常有用的。比如:
起10个线程分别取从数据库中以分页的方式去查数据,当10个线程都执行结束后,将总的查询数据写入的数据库中。这个时候如何保证所有线程都执行结束的这个时间点,就可以使用CyclicBarrier。

CyclicBarrier:同步辅助类。一组线程相互等待,全部执行结束后,会唤醒CyclicBarrier指定的同步线程,执行最终的统计工作。


public class CountThread extends Thread {

    public CountThread(int count, CyclicBarrier cyclicBarrier) {
        this.count = count;
        this.cyclicBarrier = cyclicBarrier;
    }

    private int count;
    private CyclicBarrier cyclicBarrier;

    public void run() {
        try {
            Thread.sleep(10000);
        } catch (InterruptedException e1) {
            e1.printStackTrace();
        }
        System.out.println(count);
        try {
            cyclicBarrier.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (BrokenBarrierException e) {
            e.printStackTrace();
        }

    }

}



public class SumTaskThread implements Runnable {

    @Override
    public void run() {
        System.out.println("****************汇总*******************");
    }

}



public static void main(String[] args) {
        test1();
    }

    public static void test1() {
        // 同步辅助类,当所有的10个子线程都指向结束后会指向SumTaskThread线程
        CyclicBarrier cyclicBarrier = new CyclicBarrier(10, new SumTaskThread());

        for (int i = 1; i <= 10; i++) {
            new CountThread(i, cyclicBarrier).start();
        }

        System.out.println("主线程执行结束");
    }
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics