当前位置: 代码迷 >> 综合 >> junitPerf
  详细解决方案

junitPerf

热度:22   发布时间:2024-01-14 00:42:34.0

在java开发中经常使用junit的朋友一定也会喜欢junitPerf,若你不是很了解或不是很熟悉,建议你静下心来,慢慢的把下面的文字看完,这里绝没有晦涩的语言,作者可谓循循善诱、由浅入深。

 

去年的时候开始用junitPref那时候没有写点什么一直感觉很遗憾,其实有很多的好东西在用了之后都想写点什么和大家分享,除了junitPerf还有比如json、dwr、dojo等,近来总算有时间了,便想着弥补这点缺憾。然本人不会容忍自己的写的东西是一个凑数的东西,我边开始搜集资料,试图找到更多的参考。直到我发现下面这些文字,我便告诉自己:关于junitPerf的专题我不再写了因为它真的很优秀。让我们开始吧。

 

作者:Andrew Glover   来自于developworks  http://www.ibm.com/developerworks/cn/java/j-cq11296.html

 

 

追求代码质量: JUnitPerf 进行性能测试

监控可伸缩性和性能的两个简单测试

 

级别: 中级

Andrew Glover (aglover@stelligent.com), 总裁, Stelligent Incorporated

2006 11 29

在应用程序的开发周期中,性能测试常被放到最后考虑,这并不是因为它不重要,而是因为存在这么多未知变量,很难有效地测试。在本月的 追求代码质量 系列中,Andrew Glover 使性能测试成为开发周期的一部分,并介绍了两种简单的实现方法。

在应用程序的开发中,验证应用程序的性能几乎总处于次要的地位。请注意,我强调的是验证 应用程序的性能。应用程序的性能总是 首要考虑的因素,但开发周期中却很少包含对性能的验证。

由于种种原因,性能测试常被延迟到开发周期的后期。以我的经验,企业之所以在开发过程中不包含性能测试是因为,他们不知道对于正在进行开发的应用程序要期待什么。提出了一些(性能)指数,但这些指数是基于预期负载提出的。

发生下列两种情况之一时,性能测试就成为头等大事:

  • 生产中出现显而易见的性能问题。
  • 在同意付费之前 ,客户或潜在客户询问有关性能指数的问题。

本月,我将介绍两种简单的性能测试技术,在上述两种情况中的任何一种发生前进行测试。

改进代码质量

别错过 Andrew 讨论论坛,里面有关于代码语法、测试框架以及如何编写专注于质量的代码的帮助。

JUnitPerf 进行测试

在软件开发的早期阶段,使用 JUnit 很容易确定基本的低端性能指数。JUnitPerf 框架能够将测试快速地转化为简单的负载测试,甚至压力测试。

可使用 JUnitPerf 创建两种测试类型:TimedTest LoadTest。这两种类型都基于 Decorator 设计模式并利用 JUnit suite 机制。TimedTest 为测试样例创建一个(时间)上限 —— 如果超过这个时间,那么测试失败。LoadTest 和计时器一起运行,它通过运行所需的次数(时间间隔由配置的计时器控制),在一个特定的测试用例上创建一个人工负载。

 


 

 

恰当的时限测试

JUnitPerf TimedTest 让您可以编写有相关时间限制的测试 —— 如果超过了该限度,就认为测试是失败的(即便测试逻辑本身实际上是成功的)。在测试对于业务致关重要的方法时,时限测试相比其他测试来说,在确定和监控性能指数方面很有帮助。甚至可以测试得更加细致一些,可以测试一系列方法来确保它们满足特定的时间限制。

例如,假设存在一个 Widget 应用程序,其中,特定的对于业务致关重要的方法(如 createWidget())是严格的性能限制的测试目标。假设需要对执行该 create() 方法的功能方面进行性能测试。这通常会由不同的团队使用不同的工具在开发周期的后期加以确定,这通常不能指出精确的方法。但假设决定选择早期经常测试 方法取而代之。

创建 TimedTest 首先要创建一个标准的 JUnit 测试。换言之,将对 TestCase 或其派生类进行扩展,并编写一个以 test 开头的方法,如清单 1 所示:


清单 1. 简单的 widget 测试

 

public class WidgetDAOImplTest extends TestCase {   

 private WidgetDAO dao;

 

 public void testCreate() throws Exception{

  IWidget wdgt = new Widget();

  wdgt.setWidgetId(1000);

  wdgt.setPartNumber("12-34-BBD"); 

  try{

   this.dao.createWidget(wdgt);

  }catch(CreateException e){

   TestCase.fail("CreateException thrown creating a Widget");

  }         

 }

 

 protected void setUp() throws Exception {       

  ApplicationContext context =

   new ClassPathXmlApplicationContext("spring-config.xml");     

  this.dao = (WidgetDAO) context.getBean("widgetDAO");     

 }      

}

 

由于 JUnitPerf 是一个基于装饰器的框架,为了真正地驾驭它,必须提供一个 suite() 方法并将现有的测试装饰以 TimedTestTimedTest Test 和执行该测试的最大时间量作为参数。

也可以选择传入一个 boolean 标志作为第三个参数(false),这将导致测试快速失败 —— 意味着如果超过最大时间,JUnitPerf 立即 迫使测试失败。否则,测试样例将完整运行,然后失败。区别很微妙:在一个失败的样例中,不带可选标志运行测试可以帮您了解运行总时间。传入 false 值却意味着得不到运行总时间。

例如,在清单 2 中,我在运行 testCreate() 时设定了一个两秒钟的上限。如果执行总时间超过了这个时间,测试样例将失败。由于我并未传入可选的 boolean 参数,该测试将完整运行,而不管运行会持续多久。


清单 2. 为生成 TimedTest 而实现的 suite 方法

public static Test suite() {

 long maxElapsedTime = 2000; //2 seconds

 Test timedTest = new TimedTest(

   new WidgetDAOImplTest("testCreate"), maxElapsedTime);

 return timedTest;                         

}

 

此测试通常在 JUnit 框架中运行 —— 现有的 Ant 任务、Eclipse 运行器等等,会像运行任何其他 JUnit 测试一样运行这个测试。惟一的不同是,该测试将发生在计时器的上下文中。

 


回页首

 

过度的负载测试

与在测试场景中验证一个方法(或系列方法)的时间限制正好相反,JUnitPerf 也方便了负载测试。正如在 TimedTest 中一样,JUnitPerf LoadTest 也像装饰器一样运行,它通过将 JUnit Test 和额外的线程信息绑定起来,从而模拟负载。

使用 LoadTest,可以指定要模拟的用户(线程)数量,甚至为这些线程的启动提供计时机制。JUnitPerf 提供两类 TimerConstantTimer RandomTimer。通过为 LoadTest 提供这两类计时器,可以更真实地模拟用户负载。如果没有 Timer,所有线程都会同时启动。

清单 3 是用 ConstantTimer 实现的含 10 个模拟用户的负载测试:


清单 3. 为生成负载测试而实现的 suite 方法

public static Test suite() {

 int users = 10;

 Timer timer = new ConstantTimer(100);             

 return new LoadTest(

  new WidgetDAOImplTest("testCreate"),

    users, timer);                

}

 

请注意,testCreate() 方法运行 10 次,每个线程间隔 100 毫秒启动。未设定时间限制 —— 这些方法完整运行,如果其中任何的方法执行失败,JUnit 会相应地报告失败。

 


 

 

用样式进行装饰

装饰器并不局限于单个的装饰物。例如,在 Java? I/O 中,可以为 FileInputStream 装饰上一个带 BufferedReader InputStreamReader(只要记住:BufferedReader in = new BufferedReader(new InputStreamReader(new FileInputStream("infilename"), "UTF8")))。

装饰可以有多个层次,JUnitPerf TimedTest LoadTest 也是一样。当这两个类彼此装饰时,将导致一些强制的测试场景,例如像这样的场景:在一项业务中放置了负载并应用了时间限制。或者,我们可以仅仅将之前的两个测试场景以如下方式结合起来:

  • testCreate() 方法中放置一项负载。
  • 规定每个线程必须在该时间限制内结束。

我通过为一个标准 Test 装饰上 LoadTest(由 TimedTest 装饰)应用了上述规范,清单 4 显示了其结果。


清单 4. 经装饰的负载和时限测试

public static Test suite() {

 int users = 10;

 Timer timer = new ConstantTimer(100);

 long maxElapsedTime = 2000;        

 return new TimedTest(new LoadTest(

   new WidgetDAOImplTest("testCreate"), users, timer),

     maxElapsedTime);             

}

 

正如您所看到的那样,testCreate() 方法运行 10 次(每隔 100 毫秒启动一个线程),且每个线程必须在 2 秒内完成,否则整个测试场景将失败。

 


 

 

使用注意

尽管 JUnitPerf 是一个性能测试框架,但也要先大致估计一下测试要设定的性能指数。这是由于所有由 JUnitPerf 装饰的测试都通过 JUnit 框架运行,所以就存在额外的消耗,特别是在利用 fixture 时。由于 JUnit 本身用一个 setUp 和一个 tearDown() 方法装饰所有测试样例,所以要在测试场景的整个上下文中考虑执行时间。

相应地,我经常创建使用我想要的 fixture 逻辑的测试,但也会运行一个空白测试来确定性能指数基线。这是一个大致的估计,但它必须作为基线添加到任何想要的测试限制中。

例如,如果运行一个由 fixture 逻辑(使用 DbUnit)装饰的空白测试用时 2.5 秒,那么您想要的所有测试限制都应将这一额外时间考虑在内 —— 这可以从清单 5 中的基准测试中看到:


清单 5. JUnitPerf 基准测试

public class DBUnitSetUpBenchmarkTest extends DatabaseTestCase {

 private WidgetDAO dao = null;

 

 public void testNothing(){

  //should be about 2.5 seconds

 }

 

 protected IDatabaseConnection getConnection() throws Exception {        

  Class driverClass = Class.forName("org.hsqldb.jdbcDriver");

  Connection jdbcConnection =

   DriverManager.getConnection(

     "jdbc:hsqldb:hsql://127.0.0.1", "sa", "");

  return new DatabaseConnection(jdbcConnection);

 }

 

 protected IDataSet getDataSet() throws Exception {        

  return new FlatXmlDataSet(new File("test/conf/seed.xml"));

 }

 

 protected void setUp() throws Exception {

  super.setUp();     

  final ApplicationContext context =

   new ClassPathXmlApplicationContext("spring-config.xml");     

  this.dao = (WidgetDAO) context.getBean("widgetDAO");     

 }

}

 

请注意,清单 5 的测试样例 testNothing() 什么都没做。其惟一的目的是确定运行 setUp() 方法(当然,该方法也通过 DbUnit 设置了一个数据库)的总时间。

也请记住,测试时间将依赖于机器的配置而变化,同时也依赖于在执行 JUnitPerf 测试时运行的东西而变化。我经常发现,将 JUnitPerf 测试放到它们自己的分类中有助于将它们同标准测试隔离开。这意味着,在运行一个测试时不必每次都运行 JUnitPerf 测试,例如在一个 CI 环境中签入代码。我也会创建特定的 Ant 任务,从而只在精心策划的将性能测试考虑在内的场景或环境中运行这些测试。

 


 

 

试试吧!

JUnitPerf 进行性能测试无疑是一门严格的科学,但在开发生命周期的早期,这是确定和监控应用程序代码的低端性能的极佳方式。另外,由于它是一个基于装饰器的 JUnit 扩展框架,所以可以很容易地用 JUnitPerf 装饰现有的 JUnit 测试。

想想您已经花了这么多时间来担心应用程序在负载下会怎样执行。用 JUnitPerf 进行性能测试可以为您减少担忧并节省时间,同时也确保了应用程序代码的质量。

 

参考资料

学习

  • 您可以参阅本文在 developerWorks 全球站点上的 英文原文
  • Java 理论与实践: 用动态代理进行修饰Brian GoetzdeveloperWorks2005 8 月):动态代理是构建装饰器和适配器的便利工具。
  • 可重复的系统测试Andrew GloverdeveloperWorks2006 9 月):Andrew Glover 介绍了 Cargo,这是一个开放源码框架,该框架以一种通用的风格对容器管理进行自动化。
  • Continuous Performance Testing With JUnitPerfMike Clarkdevnewz2003 7 月):Mike Clark 介绍了一种让您更加自信并减轻压力的方法,即编写自动化的性能测试,因为这些测试是不会说谎的。
  • Effective Unit Testing with DbUnitAndrew GloverOnJava2004 1 月):介绍了用 DbUnit 进行的数据库依赖性测试。
  • Groovy performance testingthediscoblog.com):了解如何将 JUnitPerf 和用 Groovy 编写的测试结合使用。
  • 追求代码质量 系列Andrew GloverdeveloperWorks):获得更多有关代码语法、测试框架以及编写专注于质量的代码的信息。
  • Java 技术专区:数百篇关于 Java 编程各个方面的文章。


获得产品和技术

  • 下载 JUnit:了解 JUnit 4 的新特性。
  • 下载 JUnitPerf:一系列用于测量性能和可伸缩性的 JUnit 测试装饰器。


讨论

  • 参与论坛讨论
  • 讨论论坛:改进您的代码质量 :作为一名专注于改进代码质量的顾问,Andrew 在这里分享他的专业知识。

 

关于作者

Andrew Glover Stelligent Incorporated 的总裁,这家公司用有效的开发人员测试策略和能够让团队在早期经常地监视代码质量的持续集成技术,帮助企业解决软件质量问题。他是 Java Testing Patterns Wiley2004 9 月)一书的合著者。

 

 

--------

JunitPerf使用简明手册

代码效率对于许多开发者而言是一个非常关心的指标。而且,代码执行效率的高低对于应用的使用者来说是感觉最直观的。它往往也是一个应用是否成功的关键性指标之一。那么是否存在那些能够反映代码执行效率的工具呢?我们可以试一试junitperf

1. 使用目的

    测试单元测试的执行效率。由于在单元测试中实际调用的就是单元代码,因此,测试单元测试的执行效率可以直接的反应实际代码的效率。Junitperf实际是junit的一个decorator,通过编写用于junitperf的单元测试,我们也可使这一测试过程自动化。当前的版本是1.9,可以从:http://www.clarkware.com/software/JUnitPerf.htm下载

2. 用法

2.1.   基本使用步骤:

-          书写junit单元测试;

-          如果单元测试执行效率比预想的低,那么书写另一个单元测试,隔离可能的代码;

-          为这个单元测试书写对应的junitperf,如果测试通过,表示隔离代码不是引起问题的原因。此时,需要重复上面的步骤。如果测试不通过,就需要重新考虑被隔离代码的实现以提高效率,重复整个过程。

2.2.   使用TimedTest,测试代码是否能在指定时间内完成工作。

TimedTest的构造函数有三个参数:要测试的Test,期望测试完成的时间,当时间到时是否等待测试完成。

-          测试单个测试方法

//创建一个suite方法,方便使用junit执行

public static Test suite(  ) {

        //创建一个TestCase,其中构造函数中传的是要测试的测试方法名

Test testCase = new StringUtilTest("testMethod");

//创建一个TestSuite

TestSuite suite = new TestSuite();

//增加一个TimedTest

    suite.addTest( new TimedTest( testCase, 2000, false));

    return suite;

}

-          测试TestSuite

//创建一个suite方法,方便使用junit执行

public static Test suite(  ) {

//创建一个TestSuite

TestSuite suite = new TestSuite();

//增加一个TimedTest

    suite.addTest( new TimedTest(new TestSuite(StringUtilTest.class), 2000, false));

    return suite;

}

 

2.3.   使用LoadTest,进行压力测试。

LoadTest用来模拟一定数量的用户并行使用单元测试。它有4种构造函数,从构造函数,可以看出LoadTest的主要的功能:

-          LoadTest(Test test, int users),创建users数目的用户,每个请求执行一次。

-          LoadTest(Test test, int users, int iterations),创建users数目的用户,每个用户请求iterations次。

-          LoadTest(Test test, int users, int iterations, Timer timer),创建users数目的用户,使用timer规定请求时延,每个用户请求iterations次。

-          LoadTest(Test test, int users, Timer timer),创建users数目的用户,使用timer规定请求时延,每个用户请求1次。

Timer接口请参见对应的JavaDoc。它的使用与TimedTest类似。例子如下:

……

Test test= new StringUtilTest(“testMethod”);

TestSuite suite= new TestSuite();

Suite.addTest( new LoadTest( test, 100, 10));

……

2.4.  组合测试,发挥decorator的潜力。

junitperf实际上是junitdecorator,从decorator的定义可以得出,应该可以使用这两个相互的decorate

-          使用TimedTest decorate LoadTest,测试满足并行指标的请求是否满足效率指标。

……

Test test= new StringUtil( “testMethod”);

TestSuite suite= new TestSuite();

Suite.addTest( new TimedTest( new LoadTest( test, 100, 10), 3000, false));

……

-          使用LoadTest decorate TimeTest,测试满足效率指标的单个请求是否满足并行指标。

……

Test test= new StringUtil( “testMethod”);

TestSuite suite= new TestSuite();

Suite.addTest( new LoadTest( new TimedTest( test, 300, false), 1000, 10));

……

2.5.  使用ant,使测试自动化。

Junit提供了自动执行unit testant task,我们可以使用这个task来使junitperf的测试自动化(因为是decorator)。但是,它又是与unit test的侧重点不相同,因此有必要区分他们。从junit的介绍中可以看出ant task主要是通过命名规则来查找要执行的测试类,因为为了区分他们,我们只需要把这两种测试的命名规则不一样即可。至于其他方面,则与junit完全相同。

3. 检查表

    以下是使用在junitperf时需要注意的问题:

-          TimedTest的测试时间并非非常准确,而是一个大致的值。对于一个测试方法而言,它所测量的时间是指:setUp、测试方法和tearDown的总时间。

-          在进行LoadTest时,请谨慎的选择请求的时延,如果时延过大会造成达不真正并行的效果。

-          使用不同的命名规则区分普通的junit

       -      junitperf并不是用来替换那些执行效率测试工具的,它仅仅应该只用来进行本地化的效率测试和更好的重构。(摘自junitperf文档)。