怎样使用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();
  }
      将对象中运行多线程任务的机制提取出来,不仅方便单元测试,而且还能使得程序之间的耦合更松,扩展性更好。比如可以毫不费力的将现在的并发任务处理器替换成线程池。

使用Mockito对异步方法进行单元测试

之前我拍着胸脯承诺要维护的我博客,因此才有了这篇文章。但是请忘记我的那些承诺,我今天要写的是关于Mockito,这是一个当你写单元测试时经常会用到的对象Mock框架。...

Mockito bean多层嵌套的部分mock使用

/** * 自动注入mock管理的bean. 解决二层和多层bean嵌套的部分mock场景。即injectProductBusiness->customerBusiness->feeBusiness...
  • dknypxt
  • dknypxt
  • 2014年07月02日 15:27
  • 3715

使用模拟对象(Mock Object)技术进行测试驱动开发

方 世明 (fangshim@cn.ibm.com), 软件工程师, EMC 方世明就职于 IBM 中国软件开发中心存储部门,从事存储设备管理软件的开发工作。 简介: 测试...

Junit测试含有‘启动新线程’这一操作的方法时瞬间结束的问题

之前写了一篇关于FutureTask的Blog: http://rainbow702.iteye.com/admin/blogs/2206301 里面的源码如下(不包含之前写好的main方法):   ...

VC启动一个新线程的三种方法

主要用AfxBeginThread()函数来 UINT  myproc(LPVOID  lParam) { CITTDlg *pWnd = (CITTDlg *)lParam; pWnd->KMe...

mock object 测试 模拟对象

术语Tested Object – 被测对象Mock – 假的 or 仿制的对象 What is Mock Object?在讨论中我大致了解到Mock Object一般是用来做辅助单元测试,它负责隔离...
  • lionzl
  • lionzl
  • 2011年02月17日 17:20
  • 462

JUnit学习笔记10---mock object进行孤立测试4

尝试2:使用类工厂进行重构       我们采用Inversion of Control(IoC)模式,这个模式要求用到的任何的资源都要传给getContent方法或WebClient类。现在我...

单元测试的扩展:mock object

现实问题:      在实际的面向对象软件设计中,我们经常会碰到这样的情况,我们在对现实对象进行构建之后,对象之间是通过一系列的接口来实现。这在面向对象设计里是最自然不过的事情了,但是随着软件测试需...

使用 Python Mock 类进行单元测试

Data type, model, or node -- these are just some of the roles a mock object might assume. But how do...

Mock测试工具类的设计与使用

测试工具前段时间的时候,在写很多单元测试,用了比较多的Mockito。 但是有个比较麻烦的事情就是需要调用很多的set方法,甚至有部分被mock的类使用了Spring的注解来注入,并没有使用set方...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:怎样使用mock object测试一个启动新线程的类
举报原因:
原因补充:

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