绝对好文, 快2点了, 可算看到点好东西.
1,为了尽量平稳过渡,继续用junit3.8,而不是直接使用junit4
2,@ContextConfiguration(locations = { "classpath:applicationContext-sys.xml",
"classpath:applicationContext.xml" })
public class BaseTests extends AbstractTransactionalJUnit38SpringContextTests {
}
所有的TestCase都继承上面这个类,使得spring配置文件重用,同时ApplicationContext也是同一个。
3,要测试的service类通过@Autowired注入,例如
@Autowired
private UserDAO userDAO;
4,注意,对于继承了BaseTests的测试类里的每一个test方法,运行测试时都会按
setUp()-->test方法-->tearDown()的顺序执行,而且这3个组成一个事务,执行完后默认会回滚。
如果不想回滚,在类名前面@TransactionConfiguration(defaultRollback = false)
5,因为还是用的junit3.8,故TestSuite的写法照旧
public class SuiteTests {
public static Test suite() {
TestSuite suite = new TestSuite("运行所有测试");
suite.addTestSuite(UserServiceTests.class);
suite.addTestSuite(RoleServiceTests.class);
return suite;
}
}
6,测试类并没有当作bean配置在application.xml文件中,
spring测试框架是如何使"setUp()-->test方法-->tearDown()"处于同一个事务的呢?
具体就要看AbstractTransactionalJUnit38SpringContextTests了,是通过注解实现的。
service方法在test方法里面,这两个方法都配置有事务,即形成了事务的嵌套,这里有点复杂了,我还没想清楚。
不过可以肯定的是,如果test方法的事务不提交,那么service方法的事务也不会提交。
7,如果想在jdk1.4下重用ApplicationContext,可采用以下的办法,出处再补上。这个办法与前面的进行比较,
我觉得最大的区别是这种写法要手动进行测试数据的清除。
public class BaseTestCase extends TestCase {
static ApplicationContext context = null;
static {
// perform the "global" set up logic
// 这里的代码会在类初始化时执行,所以相当于BeforeClass
String[] CONFIG_FILES_MIN = { "applicationContext-sys.xml",
"applicationContext.xml" };
context = new ClassPathXmlApplicationContext(CONFIG_FILES_MIN);
System.out.println("setup");
// and now register the shutdown hook for tear down logic
// 将一个匿名方法写到这里,就相当于AfterClass
Runtime.getRuntime().addShutdownHook(new Thread() {
public void run() {
System.out.println("shutdown");
}
});
}
protected void setUp() throws Exception {
}
}
8,如果用junit-4.jar,去官网下载junit4.4.jar。不要采用myeclipse自带的junit4.jar,因为这个不带JUnit4ClassRunner类。
另外不要用junit-4.5.jar,据说与spring2.5有冲突,
详见Spring Test Context Framework not compatible with JUnit 4.5,http://jira.springframework.org/browse/SPR-5145