怎样使用mock object测试一个启动新线程的类

翻译 2006年06月15日 23:23:00

       本文是在jmock的网站上发现的,很有实际意义,因为一直用easymock,试了一下jmock,觉得很别扭,方法名以字符串的方式自己输入,容易写错,而且还要继承它自己的基类,不爽。
       所以本文的程序样例用easymock重写了。

      在下面的例子中,Guard持有一个Alarm的引用,在必要的时候进行报警。

public interface Alarm {
  public void ring();
}
public class Guard {
  private Alarm alarm;

  public Guard(Alarm alarm) {
    this.alarm = alarm;
  }

  public void getBored() {
    startRingingTheAlarm();
  }

  private void startRingingTheAlarm() {
    Runnable ringAlarmTask = new Runnable() {
      public void run() {
        alarm.ring();
      }
    };
    Thread ringAlarmThread = new Thread(ringAlarmTask);
    ringAlarmThread.start();
  }
}

Guard.getBored()的测试代码如下:

public void testGuardDoesNotRingTheAlarmWhenHeGetsBored() {
    Alarm alarm = EasyMock.createMock(Alarm.class);
    Guard guard = new Guard(alarm);
    guard.getBored();
  }

      在此例中,预期的异常并没有发生,测试通过了。这是因为alarm抛出的异常是在ringAlarm线程中,而不是在测试主线程中。此问题的根源是试图使用mock object来进行集成测试。用mock object来进行单元测试是希望将测试的单元与系统其他单元相隔离。然而,线程从其特性来说,是属于集成测试的范畴。并发和同步都要涉及到全局范围,线程的创建也用到了操作系统底层的特性。

      一种解决方案是将要执行任务的对象与任务的细节相隔离,在它们之间引入一个接口。这样,可以用mock object来测试要执行任务的对象,在集成测试中测试任务的执行。

public interface TaskRunner {
  public void start(Runnable task);
}
测试方案如下:
public void testGuardDoesNotRingTheAlarmWhenHeGetsBored() {
    Alarm alarm = EasyMock.createMock(Alarm.class);
    TaskRunner taskRunner = new ImmediateTaskRunner();
    Guard guard = new Guard(alarm, taskRunner);
    guard.getBored();
  }

       在TaskRunner的实现中,如果是启用一个新线程来执行任务,那么又回到了问题的开始,测试还是不能得到希望的异常。我们需要将任务的执行放在TaskRunner相同的线程中。最简单的方法就是立即执行该任务,而不是启线程来执行。在单元测试中,可以直接实现TaskRunner接口,得到如下的任务执行器。

public class ImmediateTaskRunner implements TaskRunner {
  public void start(Runnable task) {
    task.run();
  }
}

Guard代码更新如下:

public class Guard {
  private Alarm alarm;

  private TaskRunner taskRunner;

  public Guard(Alarm alarm, TaskRunner taskRunner) {
    this.alarm = alarm;
    this.taskRunner = taskRunner;
  }

  public void getBored() {
    startRingingTheAlarm();
  }

  private void startRingingTheAlarm() {
    Runnable ringAlarmTask = new Runnable() {
      public void run() {
        alarm.ring();
      }
    };
    taskRunner.start(ringAlarmTask);
  }
}

在实际项目中使用的TaskRunner

public class ConcurrentTaskRunner implements TaskRunner {
  public void start(Runnable task) {
    (new Thread(task)).start();
  }
}
       另一种方案是在Guard.getBored()执行结束后,在测试所在的线程中执行任务。如果Guard中的try/finally 掩盖了任务引起的测试错误,应用此方案则特别适合。
实现的TaskRunner如下:
public class DelayedTaskRunner implements TaskRunner {
  private List<Runnable> delayedTasks = new ArrayList<Runnable>();

  public void start(Runnable task) {
    delayedTasks.add(task);
  }

  public void runTasks() {
    for (Iterator<Runnable> i = delayedTasks.iterator(); i.hasNext();) {
      i.next().run();
      i.remove();
    }
  }
}
对应的测试代码为:
public void testGuardDoesNotRingTheAlarmWhenHeGetsBored() {
    Alarm alarm = EasyMock.createMock(Alarm.class);
    DelayedTaskRunner taskRunner = new DelayedTaskRunner();
    Guard guard = new Guard(alarm, taskRunner);
    guard.getBored();
    taskRunner.runTasks();
  }
      将对象中运行多线程任务的机制提取出来,不仅方便单元测试,而且还能使得程序之间的耦合更松,扩展性更好。比如可以毫不费力的将现在的并发任务处理器替换成线程池。

怎样使用mock object测试一个启动新线程的类

       本文是在jmock的网站上发现的,很有实际意义,因为一直用easymock,试了一下jmock,觉得很别扭,方法名以字符串的方式自己输入,容易写错,而且还要继承它自己的基类,不爽。 ...
  • itkbase
  • itkbase
  • 2008年03月11日 16:27
  • 317

C++单元测试二:何时Mock及其是与非

什么时候需要mock 在前面一部分(C++单元测试一:并非看上去那么简单——几个很实际的问题),我遇到的问题是:一个单元测试工程只能测一个被测类(其实前文的后记部分也已指出,其实创建新工程也不是特别...
  • henan_lujun
  • henan_lujun
  • 2013年06月02日 23:01
  • 3694

junit--用mock object进行隔离测试(上)

最近抽空翻译了一下manning出版的junit in action,译的比候捷还糟,大家不要bt.以下是第7章的内容第一节   mock object 的介绍:      隔离测试有很多好处,比如测...
  • huabingl
  • huabingl
  • 2005年02月28日 21:18
  • 4437

【整理】python单元测试Mock总结

Python写单元大多数都会用到unittest和mock,测试代码覆盖率都会用到coverage,最后再用nose把所有的东西都串起来,这样每次出版本,都能把整个项目的单元测试都运行一遍。Unitt...
  • iamonlyme
  • iamonlyme
  • 2017年06月07日 22:04
  • 1319

如何创建和启动一个线程?

原文转自:http://www.tqcto.com/article/recommend/137.html一、定义线程 1、继承java.lang.Thread类。 此类中有个run()方法,应该注意其...
  • u014723529
  • u014723529
  • 2015年09月02日 13:05
  • 3311

测试工作中的 Mock 实现

转载地址:https://testerhome.com/topics/4021 实际工作中,测试角色可能会遇到如下情况: 场景一:甲开发A模块,乙开发B模块,甲的进度比乙快,但A模块的方法...
  • wanglha
  • wanglha
  • 2016年01月22日 16:54
  • 1773

说说初用 Mock 工具测试碰到的坑

说说初用 Mock 工具测试碰到的坑 我是一个在校实习生,作为一个程序猿,是个菜鸟中战斗机!对于测试,只写过一点点简单到不能再简单了的 Junit 单元测试的例子(因为当时这足以应付学校课程的内容...
  • GarfieldEr007
  • GarfieldEr007
  • 2016年12月27日 22:57
  • 2089

Android 开启一个新线程

package com.example.che;import android.os.Bundle; import android.app.Activity; import android.util.L...
  • qq_15267341
  • qq_15267341
  • 2017年02月06日 18:49
  • 1806

java开启新线程的三种方法

见:http://blog.csdn.net/caidie_huang/article/details/52748460 见:http://blog.csdn.net/typename/article...
  • u011314442
  • u011314442
  • 2017年02月28日 09:12
  • 1643

SpringMVC使用MockMvc和GroboUtils进行多线程测试

Junit本身是不支持普通的多线程测试的,这是因为Junit的底层实现上,是用System.exit退出用例执行的。JVM都终止了,在测试线程启动的其他线程自然也无法执行。所以要想编写多线程Junit...
  • MissGeng
  • MissGeng
  • 2017年02月22日 11:08
  • 731
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:怎样使用mock object测试一个启动新线程的类
举报原因:
原因补充:

(最多只允许输入30个字)