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

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

[初学者级别]

在刚开始接触 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. 创建单元测试基类

  • 使用 Junit 开始单元测试。
    a. 创建的测试类需要继承单元测试基类。
    b. 可以正常使用 Spring 的注解方式注入 service 等。
    c. 需要注意 Jeeplus 中使用的 Shiro 权限控制问题,很多 Jeeplus 框架 中的方法可能有权限控制问题。
  • eg:

    
    
  • 好吧,还是忍受不住没有 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 了。

    采用前辈推荐的方法:

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

    image.png

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

    相关帖子

    欢迎来到这里!

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

    注册 关于
    请输入回帖内容 ...
    • PeterChu
      作者

      还有个需要注意的地方就是:
      还没去看怎么配置下 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
    人生是场修行,求知是种信仰 ! 西安

    推荐标签 标签

    • 智能合约

      智能合约(Smart contract)是一种旨在以信息化方式传播、验证或执行合同的计算机协议。智能合约允许在没有第三方的情况下进行可信交易,这些交易可追踪且不可逆转。智能合约概念于 1994 年由 Nick Szabo 首次提出。

      1 引用 • 11 回帖 • 2 关注
    • SpaceVim

      SpaceVim 是一个社区驱动的模块化 vim/neovim 配置集合,以模块的方式组织管理插件以
      及相关配置,为不同的语言开发量身定制了相关的开发模块,该模块提供代码自动补全,
      语法检查、格式化、调试、REPL 等特性。用户仅需载入相关语言的模块即可得到一个开箱
      即用的 Vim-IDE。

      3 引用 • 31 回帖 • 104 关注
    • 京东

      京东是中国最大的自营式电商企业,2015 年第一季度在中国自营式 B2C 电商市场的占有率为 56.3%。2014 年 5 月,京东在美国纳斯达克证券交易所正式挂牌上市(股票代码:JD),是中国第一个成功赴美上市的大型综合型电商平台,与腾讯、百度等中国互联网巨头共同跻身全球前十大互联网公司排行榜。

      14 引用 • 102 回帖 • 354 关注
    • 大数据

      大数据(big data)是指无法在一定时间范围内用常规软件工具进行捕捉、管理和处理的数据集合,是需要新处理模式才能具有更强的决策力、洞察发现力和流程优化能力的海量、高增长率和多样化的信息资产。

      93 引用 • 113 回帖
    • 博客

      记录并分享人生的经历。

      273 引用 • 2388 回帖
    • 房星科技

      房星网,我们不和没有钱的程序员谈理想,我们要让程序员又有理想又有钱。我们有雄厚的房地产行业线下资源,遍布昆明全城的 100 家门店、四千地产经纪人是我们坚实的后盾。

      6 引用 • 141 回帖 • 584 关注
    • WebComponents

      Web Components 是 W3C 定义的标准,它给了前端开发者扩展浏览器标签的能力,可以方便地定制可复用组件,更好的进行模块化开发,解放了前端开发者的生产力。

      1 引用 • 5 关注
    • Jenkins

      Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

      53 引用 • 37 回帖 • 3 关注
    • Ant-Design

      Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

      17 引用 • 23 回帖 • 4 关注
    • 以太坊

      以太坊(Ethereum)并不是一个机构,而是一款能够在区块链上实现智能合约、开源的底层系统。以太坊是一个平台和一种编程语言 Solidity,使开发人员能够建立和发布下一代去中心化应用。 以太坊可以用来编程、分散、担保和交易任何事物:投票、域名、金融交易所、众筹、公司管理、合同和知识产权等等。

      34 引用 • 367 回帖
    • danl
      146 关注
    • 钉钉

      钉钉,专为中国企业打造的免费沟通协同多端平台, 阿里巴巴出品。

      15 引用 • 67 回帖 • 335 关注
    • App

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

      91 引用 • 384 回帖 • 2 关注
    • uTools

      uTools 是一个极简、插件化、跨平台的现代桌面软件。通过自由选配丰富的插件,打造你得心应手的工具集合。

      6 引用 • 14 回帖
    • Chrome

      Chrome 又称 Google 浏览器,是一个由谷歌公司开发的网页浏览器。该浏览器是基于其他开源软件所编写,包括 WebKit,目标是提升稳定性、速度和安全性,并创造出简单且有效率的使用者界面。

      62 引用 • 289 回帖
    • Solo

      Solo 是一款小而美的开源博客系统,专为程序员设计。Solo 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

      这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

      1435 引用 • 10056 回帖 • 489 关注
    • 设计模式

      设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

      200 引用 • 120 回帖
    • 单点登录

      单点登录(Single Sign On)是目前比较流行的企业业务整合的解决方案之一。SSO 的定义是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统。

      9 引用 • 25 回帖
    • 域名

      域名(Domain Name),简称域名、网域,是由一串用点分隔的名字组成的 Internet 上某一台计算机或计算机组的名称,用于在数据传输时标识计算机的电子方位(有时也指地理位置)。

      43 引用 • 208 回帖
    • JVM

      JVM(Java Virtual Machine)Java 虚拟机是一个微型操作系统,有自己的硬件构架体系,还有相应的指令系统。能够识别 Java 独特的 .class 文件(字节码),能够将这些文件中的信息读取出来,使得 Java 程序只需要生成 Java 虚拟机上的字节码后就能在不同操作系统平台上进行运行。

      180 引用 • 120 回帖 • 2 关注
    • Sym

      Sym 是一款用 Java 实现的现代化社区(论坛/BBS/社交网络/博客)系统平台。

      下一代的社区系统,为未来而构建

      524 引用 • 4601 回帖 • 700 关注
    • 外包

      有空闲时间是接外包好呢还是学习好呢?

      26 引用 • 232 回帖
    • 大疆创新

      深圳市大疆创新科技有限公司(DJI-Innovations,简称 DJI),成立于 2006 年,是全球领先的无人飞行器控制系统及无人机解决方案的研发和生产商,客户遍布全球 100 多个国家。通过持续的创新,大疆致力于为无人机工业、行业用户以及专业航拍应用提供性能最强、体验最佳的革命性智能飞控产品和解决方案。

      2 引用 • 14 回帖 • 2 关注
    • 正则表达式

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

      31 引用 • 94 回帖 • 2 关注
    • HBase

      HBase 是一个分布式的、面向列的开源数据库,该技术来源于 Fay Chang 所撰写的 Google 论文 “Bigtable:一个结构化数据的分布式存储系统”。就像 Bigtable 利用了 Google 文件系统所提供的分布式数据存储一样,HBase 在 Hadoop 之上提供了类似于 Bigtable 的能力。

      17 引用 • 6 回帖 • 75 关注
    • Webswing

      Webswing 是一个能将任何 Swing 应用通过纯 HTML5 运行在浏览器中的 Web 服务器,详细介绍请看 将 Java Swing 应用变成 Web 应用

      1 引用 • 15 回帖 • 637 关注
    • 阿里云

      阿里云是阿里巴巴集团旗下公司,是全球领先的云计算及人工智能科技公司。提供云服务器、云数据库、云安全等云计算服务,以及大数据、人工智能服务、精准定制基于场景的行业解决方案。

      89 引用 • 345 回帖