[jeeplus]jeeplus 中的单元测试方法论

本贴最后更新于 2201 天前,其中的信息可能已经时移世易

[初学者级别]

在刚开始接触 jeeplus 时,就一直纠结怎么用自己之前最开始的单元测试方法,对 dao 层,service 层的方法进行测试。

发现:

  1. Junit 的单元测试在 jeeplus 框架下,使用时会出现需要有 ApplicationContext ,或者需要实现模拟 请求对象等各种问题,但是 jeeplus 中的 SpringContextHolder 自己却不太会用,(或者是环境、jar 包的问题引起各种异常),不能进行常用的单元测试方法。

  2. 刚开始按照在网上找到的一些博文中的方法,也是各种尝试,同样不能解决问题。后来就搁置了好久好久。最近,实在是觉得还是非常有必要调通,会给项目的完成提供很大助力。所以,又开始找了许多博文来试、请教大佬,最终,终于调通了。

  3. 在后续的使用中,还存在着各种问题,比如 Shiro 的权限问题,还需要继续研究。

总结:

(仅简单说下大致的用法,内中详细自己还没掌握透彻,求大佬补充。)

  1. 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 -->
  1. 创建单元测试基类
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 { }
  1. 使用 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()); } }
  1. 好吧,还是忍受不住没有 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 { }

这样就可以了。最终效果:

image.png

1 操作
PeterChu 在 2019-06-06 18:22:18 更新了该帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
  • PeterChu
    作者
    package com.jeeplus.modules.junittest; import com.alibaba.fastjson.JSONObject; import com.jeeplus.core.junit.BaseJunit4Test; import com.jeeplus.modules.university.entity.SysAreaForUniversity; import com.jeeplus.modules.university.service.SysAreaForUniversityService; import org.junit.Test; import org.springframework.beans.factory.annotation.Autowired; /** * Created by pc on 2019/6/6. */ public class TestUniversity extends BaseJunit4Test { @Autowired SysAreaForUniversityService sysAreaForUniversityService; //测试 SysAreaForUniversity 中的getByNameForCity() 方法 @Test public void testGetByNameForCity(){ SysAreaForUniversity sysAreaForUniversity = new SysAreaForUniversity(); sysAreaForUniversity.setName("北京市"); SysAreaForUniversity sa = sysAreaForUniversityService.getByNameForCity(sysAreaForUniversity); System.out.println("查询到的城市:"+ sa.getName() +","+ sa.getCode()); // String jo = JSONObject.toJSONString(sa); //此处 栈溢出 异常 // System.out.println(jo); System.out.println(sa.getId()); } }

    image.png

    还有个需要注意的地方就是:
    还没去看怎么配置下 log4j 的日志输出。
    所以,现在测试时是不能查看日志的。

  • PeterChu
    作者

    单元测试中会存在 shiro 安全验证问题:

    如下图单元测试中,调用了某个 xxxService 的 findList 方法时,会报异常,但同样的情况下有时又不会报该异常。

    image.png

    org.apache.shiro.UnavailableSecurityManagerException: No SecurityManager accessible to the calling code, either bound to the org.apache.shiro.util.ThreadContext or as a vm static singleton. This is an invalid application configuration. at org.apache.shiro.SecurityUtils.getSecurityManager(SecurityUtils.java:123) at org.apache.shiro.subject.Subject$Builder.<init>(Subject.java:626) at org.apache.shiro.SecurityUtils.getSubject(SecurityUtils.java:56) at com.jeeplus.modules.sys.utils.UserUtils.getSession(UserUtils.java:290) at com.jeeplus.modules.sys.utils.UserUtils.getCache(UserUtils.java:313) at com.jeeplus.modules.sys.utils.UserUtils.getCache(UserUtils.java:308) at com.jeeplus.modules.sys.utils.UserUtils.getDataRuleList(UserUtils.java:190) at com.jeeplus.core.service.BaseService.dataRuleFilter(BaseService.java:37) at com.jeeplus.core.service.CrudService.findList(CrudService.java:61)

    暂时没有找到原因和解决办法,先记录下。

  • PeterChu
    作者
    1. 注意 @ContextConfiguration(locations={"classpath:spring/spring-context*.xml","classpath:/spring/spring-mvc*.xml"}) 路径问题,最好将 xml 配置文件按照此种方式放在 src/main/resources 目录内。
    2. 如果按照 IDEA+maven+SpringMVC 模式默认情况下,xml 文件位于 WEB-INF 目录下,那么此处应该如何获取呢?路径应该怎样写?
    3. 注意 classpath: 的使用。
    4. 在使用了该方式进行 junit 测试时,需要注意的是,如果某个组件对象中有通过注解注入属性,那么在测试类中不能使用 new 的方式创建对象。因为 new 方式创建出来的对象,由于没有经过容器创建自动注入,因此其中的属性对象为空,所以对于这类对象的创建,在测试类中同样应该使用注解的方式让容器来管理 bean 对象的创建。(问题:maven 项目默认的 test 文件夹未在 Spring 注解扫描范围内,为什么在测试类中同样可用使用注解?因为 @RunWith 还是因为 @WebAppConfiguration ?)
  • PeterChu
    作者

    回复上一条第 2 点:

    public class TestEmpDAO { ApplicationContext ac ; @Before public void before(){ // ac = new ClassPathXmlApplicationContext("dispatcher-servlet.xml"); // 这样是读取不到 WEB-INF 下的 xml 文件的 ac = new FileSystemXmlApplicationContext("web/WEB-INF/dispatcher-servlet.xml"); } }

    Junit 单元测试 Spring 读取 WEB-INF 下 xml 文件

PeterChu
人生是场修行,求知是种信仰 ! 西安

推荐标签 标签

  • 30Seconds

    📙 前端知识精选集,包含 HTML、CSS、JavaScript、React、Node、安全等方面,每天仅需 30 秒。

    • 精选常见面试题,帮助您准备下一次面试
    • 精选常见交互,帮助您拥有简洁酷炫的站点
    • 精选有用的 React 片段,帮助你获取最佳实践
    • 精选常见代码集,帮助您提高打码效率
    • 整理前端界的最新资讯,邀您一同探索新世界
    488 引用 • 384 回帖 • 4 关注
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    91 引用 • 384 回帖
  • Lute

    Lute 是一款结构化的 Markdown 引擎,支持 Go 和 JavaScript。

    29 引用 • 202 回帖 • 27 关注
  • CSDN

    CSDN (Chinese Software Developer Network) 创立于 1999 年,是中国的 IT 社区和服务平台,为中国的软件开发者和 IT 从业者提供知识传播、职业发展、软件开发等全生命周期服务,满足他们在职业发展中学习及共享知识和信息、建立职业发展社交圈、通过软件开发实现技术商业化等刚性需求。

    14 引用 • 155 回帖
  • PWL

    组织简介

    用爱发电 (Programming With Love) 是一个以开源精神为核心的民间开源爱好者技术组织,“用爱发电”象征开源与贡献精神,加入组织,代表你将遵守组织的“个人开源爱好者”的各项条款。申请加入:用爱发电组织邀请帖
    用爱发电组织官网:https://programmingwithlove.stackoverflow.wiki/

    用爱发电组织的核心驱动力:

    • 遵守开源守则,体现开源&贡献精神:以分享为目的,拒绝非法牟利。
    • 自我保护:使用适当的 License 保护自己的原创作品。
    • 尊重他人:不以各种理由、各种漏洞进行未经允许的抄袭、散播、洩露;以礼相待,尊重所有对社区做出贡献的开发者;通过他人的分享习得知识,要留下足迹,表示感谢。
    • 热爱编程、热爱学习:加入组织,热爱编程是首当其要的。我们欢迎热爱讨论、分享、提问的朋友,也同样欢迎默默成就的朋友。
    • 倾听:正确并恳切对待、处理问题与建议,及时修复开源项目的 Bug ,及时与反馈者沟通。不抬杠、不无视、不辱骂。
    • 平视:不诋毁、轻视、嘲讽其他开发者,主动提出建议、施以帮助,以和谐为本。只要他人肯努力,你也可能会被昔日小看的人所超越,所以请保持谦虚。
    • 乐观且活跃:你的努力决定了你的高度。不要放弃,多年后回头俯瞰,才会发现自己已经成就往日所仰望的水平。积极地将项目开源,帮助他人学习、改进,自己也会获得相应的提升、成就与成就感。
    1 引用 • 487 回帖 • 1 关注
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    10 引用 • 15 回帖 • 8 关注
  • 叶归
    11 引用 • 50 回帖 • 20 关注
  • Git

    Git 是 Linux Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控制软件。

    211 引用 • 358 回帖
  • React

    React 是 Facebook 开源的一个用于构建 UI 的 JavaScript 库。

    192 引用 • 291 回帖 • 372 关注
  • Markdown

    Markdown 是一种轻量级标记语言,用户可使用纯文本编辑器来排版文档,最终通过 Markdown 引擎将文档转换为所需格式(比如 HTML、PDF 等)。

    171 引用 • 1537 回帖 • 1 关注
  • 正则表达式

    正则表达式(Regular Expression)使用单个字符串来描述、匹配一系列遵循某个句法规则的字符串。

    31 引用 • 94 回帖
  • CodeMirror
    2 引用 • 17 回帖 • 167 关注
  • H2

    H2 是一个开源的嵌入式数据库引擎,采用 Java 语言编写,不受平台的限制,同时 H2 提供了一个十分方便的 web 控制台用于操作和管理数据库内容。H2 还提供兼容模式,可以兼容一些主流的数据库,因此采用 H2 作为开发期的数据库非常方便。

    11 引用 • 54 回帖 • 672 关注
  • InfluxDB

    InfluxDB 是一个开源的没有外部依赖的时间序列数据库。适用于记录度量,事件及实时分析。

    2 引用 • 96 关注
  • Thymeleaf

    Thymeleaf 是一款用于渲染 XML/XHTML/HTML5 内容的模板引擎。类似 Velocity、 FreeMarker 等,它也可以轻易的与 Spring 等 Web 框架进行集成作为 Web 应用的模板引擎。与其它模板引擎相比,Thymeleaf 最大的特点是能够直接在浏览器中打开并正确显示模板页面,而不需要启动整个 Web 应用。

    11 引用 • 19 回帖 • 396 关注
  • DevOps

    DevOps(Development 和 Operations 的组合词)是一组过程、方法与系统的统称,用于促进开发(应用程序/软件工程)、技术运营和质量保障(QA)部门之间的沟通、协作与整合。

    59 引用 • 25 回帖 • 1 关注
  • 支付宝

    支付宝是全球领先的独立第三方支付平台,致力于为广大用户提供安全快速的电子支付/网上支付/安全支付/手机支付体验,及转账收款/水电煤缴费/信用卡还款/AA 收款等生活服务应用。

    29 引用 • 347 回帖
  • 电影

    这是一个不能说的秘密。

    122 引用 • 608 回帖 • 1 关注
  • jsDelivr

    jsDelivr 是一个开源的 CDN 服务,可为 npm 包、GitHub 仓库提供免费、快速并且可靠的全球 CDN 加速服务。

    5 引用 • 31 回帖 • 106 关注
  • IBM

    IBM(国际商业机器公司)或万国商业机器公司,简称 IBM(International Business Machines Corporation),总公司在纽约州阿蒙克市。1911 年托马斯·沃森创立于美国,是全球最大的信息技术和业务解决方案公司,拥有全球雇员 30 多万人,业务遍及 160 多个国家和地区。

    17 引用 • 53 回帖 • 145 关注
  • 笔记

    好记性不如烂笔头。

    310 引用 • 794 回帖
  • 生活

    生活是指人类生存过程中的各项活动的总和,范畴较广,一般指为幸福的意义而存在。生活实际上是对人生的一种诠释。生活包括人类在社会中与自己息息相关的日常活动和心理影射。

    230 引用 • 1432 回帖
  • Mobi.css

    Mobi.css is a lightweight, flexible CSS framework that focus on mobile.

    1 引用 • 6 回帖 • 764 关注
  • 友情链接

    确认过眼神后的灵魂连接,站在链在!

    24 引用 • 373 回帖
  • 知乎

    知乎是网络问答社区,连接各行各业的用户。用户分享着彼此的知识、经验和见解,为中文互联网源源不断地提供多种多样的信息。

    10 引用 • 66 回帖
  • Quicker

    Quicker 您的指尖工具箱!操作更少,收获更多!

    37 引用 • 157 回帖
  • Ngui

    Ngui 是一个 GUI 的排版显示引擎和跨平台的 GUI 应用程序开发框架,基于
    Node.js / OpenGL。目标是在此基础上开发 GUI 应用程序可拥有开发 WEB 应用般简单与速度同时兼顾 Native 应用程序的性能与体验。

    7 引用 • 9 回帖 • 401 关注