JUnit4.8.2源码分析-4 RunNotifier与RunListener
JUnit4运行过程中,org.junit.runner.notification. RunListener和RunNotifier运用了观察者模式。
1.观察者
观察者Observer/Listener主要作用是分析各种事件并定义对应的回调接口。
比如JDK中MouseListener处理鼠标键相关的5个动作:鼠标键被按下/pressed、释放/released、单击/clicked、光标进入或离开某组件/enters or exits。java.awt.event .MouseListener的源码:
public interface MouseListener extendsEventListener {
publicvoid mouseClicked(MouseEvent e);
publicvoid mousePressed(MouseEvent e);
publicvoid mouseReleased(MouseEvent e);
publicvoid mouseEntered(MouseEvent e);
publicvoid mouseExited(MouseEvent e);
}
那么。RunListener处理測试执行的7个动作:
1. publicvoid testRunStarted(Description description)
在全部測试将要执行前的动作。如同运动会比赛前召开开幕式一样。
2. public void testStarted(Description description)
在一个測试(如@Test)開始之前的动作。
3. public void testFinished(Description description)
相应testStarted。一个測试结束后的动作。不论測试succeeds or fails。
4.public void testRunFinished(Result result)
相应testRunStarted,全部測试执行后的动作。
5.public void testIgnored(Description description)
遇到一个@Ignore測试方法是的动作。
6. public void testFailure(Failure failure)
若測试失败。调用这个动作。
7. public void testAssumptionFailure(Failure failure)
与断言不同,Assume定义了4个静态的測试条件,如assumeTrue(boolean b)等。
假设条件不满足时调用本方法。
RunListener定义了这7个空方法,地位等同于MouseAdapter,yqj2065认为不妨用abstract修饰它。
注意:回调接口的參数,用于将数据传递给上层模块。因为7个动作发生的时机不同,RunListener中使用了Description、Failure和Result封装回调接口所需的数据。
org.junit.runner.notification.Failure封装的数据有:final Description、final Throwable。
Result封装的数据有:
privateAtomicInteger fCount // the number of tests run
privateAtomicInteger fIgnoreCount// the number of tests ignored
privatefinal List<Failure> fFailures
privatelong fRunTime// milliseconds for run the entire suite
privatelong fStartTime;
Result有一些get方法,还提供了几个便利方法如
public booleanwasSuccessful() //fFailures .size()为0f返回true
另一个自带的私有内部类Listener,用于产生Result封装的数据。比如
public voidtestFinished(Description description) throws Exception {
fCount.getAndIncrement();
}
把这个代码放在testStarted中也能够。
(能够删除这些类型)
2. TextListener
详细监听器org.junit.internal.TextListener将以打印文本的形式处理7种动作。
正如我们经常使用的System.out.println()。TextListener的打印工作由一个java.io.PrintStream完毕,而该对象由System或JUnitSystem指定。顺便说明接口JUnitSystem有两个方法:exit(int i)和PrintStream out();其子类RealSystem代码
public class RealSystem implements JUnitSystem {
publicvoid exit(int code) { System.exit(code); }
publicPrintStream out() { returnSystem.out; }
}
TextListener为编写我们自己的Listener提供了一个简单的样例。(能够删除这些类型)
3. RunNotifier
被观察目标Subject/Notifier,某种事件发生或数据/状态改变后,自己主动调用doNotify()转而调用回调。RunNotifier是一个半截子的Subject,它维护一个注冊表List<RunListener>。有addListener、removeListener操作;可是它的7个fireXxx方法触发对回调接口的调用,不涉及某种事件发生或数据/状态改变。
这就是典型的二传手式委派。
真正的幕后的Subject是谁呢?
因此这个二传手是一个孤零零的类,没有子类,全部public方法都在凝视中声称为Internaluse only。
本文涉及的类型:org.junit.runner.notification.RunListener及其子类org.junit.internal.TextListener(JUnitSystem和RealSystem)、数据Description、Failure和Result、RunNotifier
涉及的设计模式:观察者模式。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步