Junit4使用实例

一、开胃示例

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

/**
 * 学生信息实体
 */
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}});
  }
}

二、固件测试

就是每个测试方法执行前和后都执行的方法,用于自动初始化和回收资源等工作。通过 @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"); // 这句不会被执
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,881评论 4 368
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,052评论 1 301
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 110,598评论 0 250
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,407评论 0 217
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,823评论 3 294
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,872评论 1 224
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,037评论 2 317
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,778评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,505评论 1 247
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,745评论 2 253
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,233评论 1 264
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,568评论 3 260
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,231评论 3 241
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,141评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,939评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,954评论 2 283
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,784评论 2 275

推荐阅读更多精彩内容