[初学者级别]
在刚开始接触 jeeplus 时,就一直纠结怎么用自己之前最开始的单元测试方法,对 dao 层,service 层的方法进行测试。
发现:
-
Junit 的单元测试在 jeeplus 框架下,使用时会出现需要有 ApplicationContext ,或者需要实现模拟 请求对象等各种问题,但是 jeeplus 中的
SpringContextHolder
自己却不太会用,(或者是环境、jar 包的问题引起各种异常),不能进行常用的单元测试方法。 -
刚开始按照在网上找到的一些博文中的方法,也是各种尝试,同样不能解决问题。后来就搁置了好久好久。最近,实在是觉得还是非常有必要调通,会给项目的完成提供很大助力。所以,又开始找了许多博文来试、请教大佬,最终,终于调通了。
-
在后续的使用中,还存在着各种问题,比如 Shiro 的权限问题,还需要继续研究。
总结:
(仅简单说下大致的用法,内中详细自己还没掌握透彻,求大佬补充。)
- pom.xml 中需要检查是否引用了正确的 jar 包,尤其注意一些环境、jar 包的版本问题。
另外,需要注意可能还有其他 jar 的问题。
<!-- TEST begin --> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>javax.el</groupId> <artifactId>javax.el-api</artifactId> <version>2.2.4</version> </dependency> <dependency> <groupId>org.glassfish.web</groupId> <artifactId>javax.el</artifactId> <version>2.2.4</version> </dependency> <dependency> <groupId>org.apache.tomcat.embed</groupId> <artifactId>tomcat-embed-websocket</artifactId> <version>7.0.52</version> <scope>test</scope> </dependency> <!-- TEST end -->
- 创建单元测试基类
package com.jeeplus.core.junit;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.web.WebAppConfiguration;
/**
* Created by pc on 2019/6/3.
*/
@RunWith(SpringJUnit4ClassRunner.class)//表示整合JUnit4进行测试
@ContextConfiguration(locations={"classpath:spring/spring-context*.xml","classpath:/spring/spring-mvc*.xml"})
@WebAppConfiguration //声明以 web 形式进行测试
public class BaseJunit4Test {
}
- 使用 Junit 开始单元测试。
a. 创建的测试类需要继承单元测试基类。
b. 可以正常使用 Spring 的注解方式注入 service 等。
c. 需要注意 Jeeplus 中使用的 Shiro 权限控制问题,很多 Jeeplus 框架 中的方法可能有权限控制问题。
eg:
package com.jeeplus.modules.junittest;
import com.jeeplus.common.utils.IdGen;
import com.jeeplus.common.utils.SpringContextHolder;
import com.jeeplus.core.junit.BaseJunit4Test;
import com.jeeplus.modules.importxlsx.entity.yifenyiduanbiao.DYifenyiduanbiao;
import com.jeeplus.modules.importxlsx.entity.zhuanyexinxibiao.DZhuanyexinxibiao;
import com.jeeplus.modules.importxlsx.service.zhuanyexinxibiao.DZhuanyexinxibiaoService;
import com.jeeplus.modules.levelformajorsfordetails.entity.LevelForMajorsDetails;
import com.jeeplus.modules.levelformajorsfordetails.entity.LevelForMajorsForDetails;
import com.jeeplus.modules.levelformajorsfordetails.mapper.LevelForMajorsDetailsMapper;
import com.jeeplus.modules.levelformajorsfordetails.service.LevelForMajorsDetailsService;
import com.jeeplus.modules.levelformajorsfordetails.service.LevelForMajorsForDetailsService;
import com.jeeplus.modules.sys.entity.User;
import org.apache.ibatis.annotations.Param;
import org.junit.Before;
import org.junit.Test;
import org.springframework.beans.factory.annotation.Autowired;
import javax.persistence.Basic;
import java.util.Date;
import java.util.List;
/**
* Created by pc on 2019/6/3.
*/
public class JunitTestERecode extends BaseJunit4Test{
@Autowired
LevelForMajorsDetailsService levelForMajorsDetailsService;
@Autowired
LevelForMajorsForDetailsService levelForMajorsForDetailsService;
@Autowired
DZhuanyexinxibiaoService dZhuanyexinxibiaoService;
/**
* 需要注意的是:
* shiro 的权限问题,Service 中大多数查询方法都有权限控制。测试时会遇到 shiro 异常。
*/
@Before
public void setDate(){
}
//测试 LevelForMajorsDetails 中的按更新时间删除记录
@Test
public void testDeleteERecord(){
Date date = new Date();
LevelForMajorsDetails levelForMajorsDetails = new LevelForMajorsDetails();
levelForMajorsDetails.setUpdateDate(date);
levelForMajorsDetails.setId(IdGen.uuid());
levelForMajorsDetails.setIsNewRecord(true);
if(levelForMajorsDetails != null){
System.out.println("date:"+ date);
levelForMajorsDetailsService.save(levelForMajorsDetails);
System.out.println("保存成功!");
} else {
System.out.println("levelForMajorsDetails 为null");
}
List<LevelForMajorsDetails> LevelForMajorsDetailsSaved = levelForMajorsDetailsService.findByUpdate(levelForMajorsDetails);
for(LevelForMajorsDetails l : LevelForMajorsDetailsSaved){
System.out.println("测试 getUpdateDate 方法中按时间获取到的记录的时间:"+l.getUpdateDate());
}
LevelForMajorsDetails newLevelForMajorsDetails = new LevelForMajorsDetails();
newLevelForMajorsDetails.setUpdateDate(date);
System.out.println("delect 的 date:"+ date);
levelForMajorsDetailsService.deleteERecord(newLevelForMajorsDetails);
System.out.println("执行了删除!");
}
//测试 LevelForMajorsForDetails 中的按更新时间删除记录
@Test
public void testLevelForMajorsDetailsServiceGet(){
Date date = new Date();
LevelForMajorsForDetails levelForMajorsForDetails = new LevelForMajorsForDetails();
levelForMajorsForDetails.setUpdateDate(date);
if(levelForMajorsForDetails != null){
System.out.println("date:"+ date);
levelForMajorsForDetailsService.save(levelForMajorsForDetails);
System.out.println("保存成功!");
} else {
System.out.println("levelForMajorsDetails 为null");
}
List<LevelForMajorsForDetails> LevelForMajorsForDetailsSaved =
levelForMajorsForDetailsService.findByUpdate(levelForMajorsForDetails);
for(LevelForMajorsForDetails l : LevelForMajorsForDetailsSaved){
System.out.println("测试 getUpdateDate 方法中按时间获取到的记录的时间:"+l.getUpdateDate());
}
LevelForMajorsForDetails newLevelForMajorsForDetails = new LevelForMajorsForDetails();
newLevelForMajorsForDetails.setUpdateDate(date);
System.out.println("delect 的 date:"+ date);
levelForMajorsForDetailsService.deleteERecord(newLevelForMajorsForDetails);
System.out.println("执行了删除!");
}
//测试 dZhuanyexinxibiaoService 中的 setAlreadyTransform()方法
@Test
public void testDZhuanyexinxibiaoService(){
int count = dZhuanyexinxibiaoService.setAlreadyTransform();
System.out.println("成功修改了"+ count +"条记录!");
}
//测试添加 beiyong20 = "" 时的查询结果是否为查询到数据库中 beiyong20=null 的记录
@Test
public void testFindListByBeiyong20(){
DZhuanyexinxibiao dZhuanyexinxibiao = new DZhuanyexinxibiao();
dZhuanyexinxibiao.setBeiyong20("");
List<DZhuanyexinxibiao> dZhuanyexinxibiaoList = dZhuanyexinxibiaoService.findListByBeiyong20(dZhuanyexinxibiao);
System.out.println(dZhuanyexinxibiaoList.size());
}
}
- 好吧,还是忍受不住没有 Log4j 了,找了篇大佬博文,照着配置下。
原博文: Junit 单元测试使用 log4j 输出日志
Junit+spring+log4j 整合之所以麻烦,是因为 spring 与 log4j 的整合,是放在 web.xml 里的,随 tomcat 启动后,spring 才会加载 log4j,而用 junit 测试是不需要 tomcat 启动的,所以 Junit 与 log4j 的整合才比较费劲。Junit 使用 spring 时,若 spring 没加载到 log4j 就会报以下警告:
1. log4j:WARN No appenders could be found for logger(org.springframework.test.context.junit4.SpringJUnit4ClassRunner). 2. log4j:WARN Please initialize the log4j system properly. 3. log4j:WARN See http://logging.apache.org/log4j/1.2/faq.html#noconfig for more info.
推荐方法
新建 JUnit4ClassRunner 类:1. public class JUnit4ClassRunner extends SpringJUnit4ClassRunner { 2. static { 3. try { 4. Log4jConfigurer.initLogging("classpath:com/config/log4j.properties"); 5. } catch (FileNotFoundException ex) { 6. System.err.println("Cannot Initialize log4j"); 7. } 8. } 9. public JUnit4ClassRunner(Class<?> clazz) throws InitializationError { 10. super(clazz); 11. } 12. }
引用此类:
1. @RunWith(JUnit4ClassRunner.class) 2. @ContextConfiguration(locations = "classpath:com/config/springConfig.xml") 3. @Transactional 4. @TransactionConfiguration(transactionManager = "transactionManager", defaultRollback = true) 5. public class TestHibernate { 6. ... 7. }
这样,在启动 Junit 测试时,spring 就会加载 log4j 了。而且保持了灵活性。
PS:Junit 加载 spring 的 runner(SpringJUnit4ClassRunner)要优先于 spring 加载 log4j,因此采用普通方法,无法实现 spring 先加载 log4j 后被 Junit 加载。所以我们需要新建 JUnit4ClassRunner 类,修改 SpringJUnit4ClassRunner 加载 log4j 的策略。这样加载 log4j 就会优先于加载 spring 了。
采用前辈推荐的方法:
package com.jeeplus.core.junit;
import org.junit.runners.model.InitializationError;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.util.Log4jConfigurer;
import java.io.FileNotFoundException;
/**
* Created by pc on 2019/6/6.
*/
public class MyJUnit4ClassRunner extends SpringJUnit4ClassRunner {
static {
try {
Log4jConfigurer.initLogging("classpath:properties/log4j.properties");//F:\pc\CEDS 本地SVN\jeeplus\src\main\resources\properties\log4j.properties properties/log4j.properties
} catch (FileNotFoundException ex) {
System.err.println("Cannot Initialize log4j");
}
}
public MyJUnit4ClassRunner(Class<?> clazz) throws InitializationError {
super(clazz);
}
}
修改原 BaseJunit4Test 中的 @RunWith 注解中类
package com.jeeplus.core.junit;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.web.WebAppConfiguration;
/**
* Created by pc on 2019/6/3.
*/
@RunWith(MyJUnit4ClassRunner.class)//表示整合JUnit4进行测试
@ContextConfiguration(locations={"classpath:spring/spring-context*.xml","classpath:/spring/spring-mvc*.xml"})
@WebAppConfiguration //声明以 web 形式进行测试
public class BaseJunit4Test {
}
这样就可以了。最终效果:
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于