根据项目的启动的,我们有两种方式启动,一种是使用main函数的启动方式,还有一种是@Test启动,main函数的启动,我们是知道如何启动的,因为jvm指令可以执行,编译后mian函数,会创建main线程来执行main函数,并触发JVM的一系列指令执行,把代码run起来。
那@Test注解启动是如何启动的?
我们来看下面这部分代码:
@Test
public void testMethod1() {
System.out.println("in the testMethod1");
}
进行启动后,debug,截图如下:
发现启动的还是main函数,com.intellij.rt.junit.JUnitStarter#main,只不过这个是插件进行识别进行加载的,源代码idea是看不到的,github可以查看源代码,下面来分析一下@Test框架的启动过程
麻雀虽小,五脏俱全,好好解剖一番
- com.intellij.rt.junit.JUnitStarter#main
- 下面是开始执行的第一个环节,看到以下代码可以看到每个子方法做的事情很明确,一眼就能看出来要做扫描,代码之前执行动作都是有空行,阅读性很强,方便以后查看及改进
- 查看prepareStreamsAndStart方法,这里代码比较优雅的处理类调用方法,不会把需要执行的类,进行方法的传递,而是使用getAgentClass的方法进行解耦
- 好了JUnitStarter类相关的执行调用已经结束了,接下到下一站点,JUnit4IdeaTestRunner.startRunnerWithArgs执行,代码比较多,下面就贴一部分java代码, testRunner,获取到@RunWith的执行器
- 下面进入核心环节,org.junit.runner.JUnitCore#run(org.junit.runner.Runner),关于RunListener是监听相关,没有添加,就不做解释了,主要是看runner.run(notifier), runner如果没有设置@RunWith的方法,那就会运行自己的JUnit4ClassRunner执行器
- 进入执行器之后,继续查看,在ParentRunner类中的run方法,这边做了junit套件对象的构建及执行
- 下面看看套件里面如何执行,如果有beforeClass的注解则构建套件
- 下面是具体代码
- invokeExploseively就是要执行的代码,执行beforeClass的代码,next游标进行下个类的执行,下面就执行所有@Test进行执行,具体的下面如下,其中MethodInoker我们可以通过覆盖这个类BlockJUnit4ClassRunner的相同的方法,重新构造套件进行执行
- 通过该方法获取到相应的注解,执行相应的策略org.junit.runners.model.TestClass#getAnnotatedMethods(java.lang.Class<? extends java.lang.annotation.Annotation>)
- 主要核心流程大致分析了一下,细节代码后续再补充