最近在学习java多线程通信的时候,我在Junit的单元测试方法中开启了两个线程,但点击执行后程序一下子就运行结束了,只有主线程中的system.out.printf() 打印了内容,在另外两个开启的线程并没有执行,然后直接百度,才知道了原因:
转自:https://segmentfault.com/a/1190000003762719
其实junit是将test作为参数传递给了TestRunner的main函数。并通过main函数进行执行。
test函数在main中执行。如果test执行结束,那么main将会调用System.exit(0);即使还有其他的线程在运行,main也会调用System.exit(0);
System.exit()是系统调用,通知系统立即结束jvm的运行,即使jvm中有线程在运行,jvm也会停止的。所以会出现之前的那种情况。其中System.exit(0);的参数如果是0,表示系统正常退出,如果是非0,表示系统异常退出。
junit TestRunner 部分源代码:
junit.textui.TestRunner
public static void main (String[] args) {
TestRunner aTestRunner = new TestRunner();
try {
TestResult r = aTestRunner.start(args);
if (!(r.wasSuccessful()))
System.exit (1);
System.exit(0);
} catch (Exception e) {
System.err.println(e.getMessage());
System.exit(2);
}
}
如果想要在junit中开启线程,那么必须要让主线程在其他线程执行完之后才结束,解决方法有两种:
- 在主线程中使用Thread.sleep(),让主线程休眠一段时间,但需要保证休眠结束后其他线程都已经执行完毕。
- 另外一种方法是用一个线程计数器 CountDownLatch ,每执行完成一条线程,调用countDown()使计数器减1,主线程中使用CountDownLatch 的await()方法等待.