Spring单元测试

时间:2022-10-22 16:32:27

1.基于AbstractDependencyInjectionSpringContextTests

  Spring的单元测试可以使用AbstractDependencyInjectionSpringContextTests做测试,不支持事务。AbstractTransactionalDataSourceSpringContextTests是支持事务的。

  Spring单元测试需要注意的地方:

  (1)需要注入的类内成员如果不是public类型,需要定义setter方法才能正常注入(针对在Spring配置中不重复的类,如果重复出现则需要使用getBean方法);

  (2)需要引入Spring-test.jar到工程,才能继承该测试类;

  (3)默认的classpath在WEB-INF/classes/目录下,如果Spring配置文件不在该目录下需要指定位置(设置classpath或者使用项目根目录);

  (4)如果测试类比较多,可以使用一个公共的测试基类完成注入的工作。

  (5) 需要一个getConfigLocations方法指定配置文件路径。

public class Test extends AbstractDependencyInjectionSpringContextTests {
@Autowired
private PersonService personService;
public PersonService getPersonService() {
return personService;
} protected String[] getConfigLocations() {
String[] configs = new String[] { "/Iocbean.xml"};
return configs;
} public void setPersonService(PersonService personService) {
this.personService = personService;
} public void test(){
personService.save();
}
}

 2.基于JUnit的测试

  单元测试目前主要的框架包括 Junit、TestNG,还有些 MOCK 框架,例如 Jmock、Easymock、PowerMock 等。 

public class AOPTestJunit {
private AService aService;
private ApplicationContext ctx; public AService getaService() {
return aService;
} public void setaService(AService aService) {
this.aService = aService;
} @Before
public void setUp() throws Exception {
ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
} @After
public void tearDown() throws Exception {
} @Test
public void test() {
System.out.println("SpringTest JUnit test");
aService = (AService) ctx.getBean("aService");
aService.fooA("JUnit test fooA");
aService.barA();
} }

  3.基于注解的方式

  第2中方法的测试类中,我们还不能使用 Spring 的注解方式,会出现空指针异常,要实现注解方式的自动注入要使用如下的方式。 

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("/spring-jms.xml")
@Transactional
public class ProducerConsumerTest { @Autowired
private ProducerService producerService;
@Autowired
@Qualifier("queueDestination")
private Destination destination; @Test
public void testSend() {
for (int i=0; i<2; i++) {
producerService.sendMessage(destination, "你好,生产者!这是消息:" + (i+1));
}
} } 

对标签的解释:

@RunWith 注释标签是 Junit 提供的,用来说明此测试类的运行者,这里用了 SpringJUnit4ClassRunner,这个类是一个针对 Junit 运行环境的自定义扩展,用来标准化在 Spring 环境中 Junit的测试用例,例如支持的注释标签。

@ContextConfiguration 注释标签是 Spring test context 提供的,用来指定 Spring 配置信息的来源,支持指定 XML 文件位置或者 Spring 配置类名,这里我们指定 classpath 下的 /spring-jms.xml 为配置文件的位置。

@Transactional 注释标签是表明此测试类的事务启用,这样所有的测试方案都会自动的 rollback,即不用自己清除自己所做的任何对数据库的变更了。

一些常见的基于 Junit4 的注释标签在 Spring 测试环境中的使用方法:

@Test(expected=...)

此注释标签的含义是,这是一个测试,期待一个异常的发生,期待的异常通过 xxx.class 标识。例如,我们修改 AccountService.Java 的 insertIfNotExist 方法,对于传入的参数如果为空,则抛出 IllegalArgumentException,如下:

public void insertIfNotExist(Account account) {

if(account==null)

throw new IllegalArgumentException("account is null");

Account acct = accountDao.getAccountById(account.getId());

if(acct==null) {

log.debug("No "+account+" found,would insert it.");

accountDao.saveAccount(account);

}

acct = null;

}

然后,在测试类中增加一个测试异常的方法,如下:

@Test(expected=IllegalArgumentException.class)

public void testInsertException() {

service.insertIfNotExist(null);

}

运行结果是 green bar。

@Test(timeout=...)

可以给测试方法指定超时时间(毫秒级别),当测试方法的执行时间超过此值,则失败。

比如在 AccountService 中增加如下方法:

public void doSomeHugeJob() {

try {

Thread.sleep(2*1000);

} catch (InterruptedException e) {

}

}

上述方法模拟任务执行时间 2 秒,则测试方法如下:

@Test(timeout=3000)

public void testHugeJob() {

service.doSomeHugeJob();

}

上述测试方法期待 service.doSomeHugeJob 方法能在 3 秒内结束,执行测试结果是 green bar。

@Repeat

通过 @Repeat,您可以轻松的多次执行测试用例,而不用自己写 for 循环,使用方法:

@Repeat(3)

@Test(expected=IllegalArgumentException.class)

public void testInsertException() {

service.insertIfNotExist(null);

}

这样,testInsertException 就能被执行 3 次。