zoukankan      html  css  js  c++  java
  • Java魔法堂:JUnit4使用详解

    摘自http://www.cnblogs.com/fsjohnhuang/p/4061902.html

    Java魔法堂:JUnit4使用详解

    目录                                                                                             

    1. 开胃示例

    2. 固件测试

    3. 忽略测试用例

    4. 异常测试

    5. 超时测试

    6. 测试运行器

    7. 参数化测试

    8. 套件测试

    9. JUnit4.4assertThat断言

    10. assumeThat断言

    11. 参考

    一、开胃示例                            

    被测试类——获取学生数据:

    复制代码
    /**
     * 学生信息实体
     */
    public class EStudent{
      String name;
       int age;
       // 一堆getter、setter
    
       /**
        * 注意:自定义类时要重写equals方法,否则默认equals方法会使用==方式来比较对象的内存地址是否相同,而不是内容是否相同。
        */
       @Override
       public boolean equals(Object obj){
         if (obj == null || obl.getClass() != this.getClass()) return false;
         
         EStudent other = (EStudent)obj;
         if (!this.getName().equals(other.getName())
                  || this.getAge() != other.getAge()) return false;
         return true;
       }
    }
    
    /**
      * 学生数据操作接口
      */
    public interface StudentDao{
      public EStudent getStudent(int id);
    }
    
    /**
     * 真实的数据操作类
     */
    public class StudentDaoImpl 
      implements StudentDao{
    
      public EStudent getStudent(int id){
           EStudent ret = new EStudent();
           ret.setName("fsjohnhuang");
           ret.setAge(18);
        
           return ret;
      }
    }
    
    /**
     * 模拟的数据操作类
     */
    public class MockStudentDaoImpl 
      implements StudentDao{
    
      public EStudent getStudent(int id){
           EStudent ret = new EStudent();
           ret.setName("fsjohnhuang");
           ret.setAge(18);
        
           return ret;
      }
    }
    复制代码

    测试用例:

    复制代码
    /**
     * 由于下面采用参数化测试,因此将测试运行器更换为Parameterized
     */
    @RunWith(Parameterized.class)
    public class TestStudentDao{
      static StudentDao dao, mockDao;
    
      /**
       * 执行测试类中所有测试前执行一次
       * 用于前提条件的初始化
       */
      @BeforeClass
      public static void init(){
          dao = new StudentDaoImpl();
          mockDao = new MockStudentDaoImpl();
      }
      
      /**
       * 接收测试用数据
       */
      int id;
      public TestStudentDao(int id){
        this.id = id;
      }
    
      /**
       * 测试用例
       */
      @Test
      public void testGetStudent(){
        assertEquals("获取学生信息", mockDao.getStudent(id), dao.getStudent(id));
      }
    
      /**
       * 测试用数据提供方法
       */
      @Parameters
      public Collection dataFeed(){
        return Arrays.asList(new Object[][]{{1},{2},{3}});
      }
    }
    复制代码

    第一次接触时可能不能理解上面的代码片段到底发生了什么事,ctrl+c到eclipse那执行一下吧!下面我们详细了解JUnit4吧!

    二、固件测试                                

      就是每个测试方法执行前和后都执行的方法,用于自动初始化和回收资源等工作。通过 @Before 注解标注测试方法执行前调用的固件测试,通过 @After 注解标注测试方法执行后调用的固件测试。

      父类的固件测试会在子类的固件测试前被调用。

      另外可通过注解 @BeforeClass 和 @AfterClass 标注某些static方法为测试开始前和结束后被调用,用于自动初始化和回收资源等工作。注意通过注解 @BeforeClass 和 @AfterClass 标注的方法一次测试过程仅被调用一次而已。示例如下:

    复制代码
    public class MyUT{
      int i, j;
      @BeforeClass public void static init(){
        System.out.println("init");
        i = 0;
        j = 0;
      }
      
      @Before public void invokeBefore(){
        System.out.println("invokeBefore" + ++i);
      }
    
      @Test public void testMyMethod1(){
        System.out.println("testMyMethod1");
      }
    
      @Test public void testMyMethod2(){
        System.out.println("testMyMethod2");
      }
    
      @After public void invokeAfter(){
        System.out.println("invokeAfter" + ++j);
      }
    
      @AfterClass public void static destroy(){
        System.out.println("destroy");
        i = 0;
        j = 0;
      }
    }
    
    // 输出结果
    init
    invokeBefore1
    testMyMethod1
    invokeAfter1
    invokeBefore2
    testMyMethod2
    invokeAfter2
    destroy
    复制代码

    三、忽略测试用例                              

      通过注解 @Ignore() 可以标注不参与测试的测试方法。当然也可以通过去除注解 @Test 来达到这个目的,但去除注解 @Test 会令到eclipse的JUnit View中无法显示该测试方法。

    四、异常测试                                

      通过注解 @Test(expected=Class类型) 来标注期待测试方法执行时抛出哪种异常对象,若测试方法不抛出异常或异常对象与期待的异常对象不相同则测试失败。

    @Test(expected=ArithmeticException.class) 
    public void calc(){
      int i = 1/0;
    }

    五、超时测试                                

      通过注解 @Test(timeout=毫秒数) 来标注期待执行测试方法的最大耗时,若超时则测试失败。

    @Test(timeout=1000) 
    public void wait(){
      while(true){}
    }

    六、测试运行器                               

      用于执行JUnit中所有的测试方法。JUnit为单元测试提供默认的测试运行器,但我们可以自定义,自定义的测试运行器必须继承 org.junit.runner.Runner 。然后通过类注解 @RunWith(CustomTestRunner.class) 来指定该测试的测试运行器。

    常用的内置测试运行器:

      1.  org.junit.runners.Suite ,套件测试时使用。

      2.  org.junit.runners.Parameterized ,参数化测试时使用。

    七、参数化测试                               

      就是第一节中的测试类型,用于模拟以不同的参数组合来对方法进行多次测试。若不使用参数化测试,该测试方法有N个不同的参数组合,则需要写N个测试方法来测试。

    复制代码
    // 需要使用Parameterized测试运行器才可以
    @RunWith(Parameterized.class)
    public class MyUT{
      // 成员变量,用于存放测试用数据和测试期望值
      int orig;
      int expected;
      public MyUT(int orig, int expected){
        this.orig = orig;
        this.expected = expected;
      }
      
      @Test public void testMyMethod(){
          assertEquals(expected, orig + 1);
      }
    
      /**
       * 测试数据和测试期望值的提供方法
       * 必须用注解@Parameters标注
       * 必须返回Collection类型数据
       */
      @Parameters public Collection dataFeed(){
        return Arrays.asList(new Object[][]{
          {1, 2},
          {2, 3},
          {3, 4}
        });
      }
    }
    复制代码

    八、套件测试                              

      JUnit4去除JUnit3中套件测试注解,而是通过另一形式提供套件测试。

    套件测试就是按业务逻辑将测试类进行分组,并以组为单位执行单元测试。

    复制代码
    // 测试类1
    public class MyUT1{
      @Test public void testMyMehthod1(){
        assertEquals(1,1);
      }
    }
    // 测试类2
    public class MyUT2{
      @Test public void testMyMehthod2(){
        assertEquals(2,2);
      }
    }
    // 套件测试类
    @RunWith(Suite.class)
    @SuiteClass({MyUT1.class, MyUT2.class})
    public class SuiteTest{
      // 必须有一个public,无参数的构造函数。使用默认的构造函数也可以
      public SuiteTest(){}
    }
    复制代码

    九、JUnit4.4的 assertThat断言                       

      JUnit4.4内置Hamcrest测试组件的部分内容,而 assertThat断言 就是配置Hamcrest测试组件的匹配符来实现所有测试工作。由于Hamcrest的匹配符贴近自然语言,因此意思表达更明确。(JUnit4.4前的版本则需要引入hamcrest-core.jar和hamcrest-library.jar了)。

    复制代码
    /* assertThat语法
     * assertThat(T actual, Matcher<T> matcher);
     * assertThat(String reason, T actual, Matcher<T> matcher);
     * 入参actual为实际值,入参matcher为期待值的匹配符
     */
    
    //测试变量是否大于指定值
    assertThat(1, greaterThan(50));
    //测试变量是否小于指定值
    assertThat(1, lessThan(100));
    //测试变量是否大于等于指定值
    assertThat(1, greaterThanOrEqualTo(50));
    //测试变量是否小于等于指定值
    assertThat(1, lessThanOrEqualTo(100));
                      
    //测试所有条件必须成立
    assertThat(1, allOf(greaterThan(50),lessThan(100)));
    //测试只要有一个条件成立
    assertThat(1, anyOf(greaterThanOrEqualTo(50), lessThanOrEqualTo(100)));
    //测试无论什么条件成立(还没明白这个到底是什么意思)
    assertThat(1, anything());
    //测试变量值等于指定值
    assertThat(1, is(100));
    //测试变量不等于指定值
    assertThat(1, not(50));
    
    /**字符串**/
    String url = "http://www.taobao.com";
    //测试变量是否包含指定字符
    assertThat(url, containsString("taobao"));
    //测试变量是否已指定字符串开头
    assertThat(url, startsWith("http://"));
    //测试变量是否以指定字符串结尾
    assertThat(url, endsWith(".com"));
    //测试变量是否等于指定字符串
    assertThat(url, equalTo("http://www.taobao.com"));
    //测试变量再忽略大小写的情况下是否等于指定字符串
    assertThat(url, equalToIgnoringCase("http://www.taobao.com"));
    //测试变量再忽略头尾任意空格的情况下是否等于指定字符串
    assertThat(url, equalToIgnoringWhiteSpace("http://www.taobao.com"));
    
    /**集合**/
    List<User> user = new ArrayList<User>();
    user.add(test1);
    user.add(test2);
                      
    //测试集合中是否还有指定元素
    assertThat(user, hasItem(test1));
    assertThat(user, hasItem(test2));
      
    /**Map匹配**/
    Map<String,User> userMap = new HashMap<String,User>();
    userMap.put(test1.getUsername(), test1);
    userMap.put(test2.getUsername(), test2);
                      
    //测试map中是否还有指定键值对
    assertThat(userMap, hasEntry(test1.getUsername(), test1));
    //测试map中是否还有指定键
    assertThat(userMap, hasKey(test2.getUsername()));
    //测试map中是否还有指定值
    assertThat(userMap, hasValue(test2));
    复制代码

    十、 assumeThat假设断言                             

      位于 org.junit.Assume类 下,同样是属于Hamcrest组件的。用于假设当条件成立时才会执行后续的代码,条件不成立时是不会影响测试结果。

    assumeThat(1, is(0));
    System.out.println("I'm here"); // 这句不会被执

      尊重原创,转载请注明来自:http://www.cnblogs.com/fsjohnhuang/p/4061902.html  ^_^肥仔John

    十一、参考                                

    http://blog.csdn.net/skyie53101517/article/details/8739126

  • 相关阅读:
    random 模块
    re 模块
    正则表达式
    15. 3Sum
    253. Meeting Rooms II
    91. Decode Ways
    17. Letter Combinations of a Phone Number
    314. Binary Tree Vertical Order Traversal
    311. Sparse Matrix Multiplication
    311. Sparse Matrix Multiplication
  • 原文地址:https://www.cnblogs.com/LiuYanYGZ/p/6187574.html
Copyright © 2011-2022 走看看