• 用junitperf做并发测试带来的问题


    关于junitperf的一点介绍

    junitperf 是个很小巧的java性能测试框架,可以在http://sourceforge.net/project/showfiles.php?group_id=15278 上下载到。可以很容易的把它结合junit一起测试,比如在测试套件里面加这么几行:

    import com.clarkware.junitperf.TimedTest;
    import com.clarkware.junitperf.LoadTest;
    .......

        int testTimes = 10;
        int users = 5;

        suite.addTest(new TestUserDAO("test1AddUser")); //基本功能测试,同时初始化环境
      
        suite.addTest(new TimedTest(new TestUserDAO("test1AddUser"), 1000)); //基本性能测试:方法应在1秒内完成

        suite.addTest(new LoadTest(new TestUserDAO("test1AddUser"), users)); // 并发测试

      suite.addTest(new LoadTest(new TestUserDAO("test1AddUser"),users,testTimes)); //并发负载测试

      suite.addTest(new TimedTest(new LoadTest(new TestUserDAO("test1AddUser"),
                                               users,testTimes), 35000)); //并发性能测试

    如果只需要反复做一个测试而不需要并发测试,可以

    suite.addTest(new LoadTest(new TestUserDAO("test1AddUser"),1,testTimes));

    当然也可以不用junitperf,junit.extensions.RepeatedTest就是设计来干这个的:

    suite.addTest(new RepeatedTest(new TestUserDAO("test1AddUser"), testTimes)); //重复测试

    用junitperf做并发测试带来的问题

    做并发测试的时候junitperf有一个问题。注意看这一行:

    new LoadTest(new TestUserDAO("test1AddUser"), users)

    我们只传递了一个TestUserDAO实例给LoadTest,却要求它开启users个线程来测试,这样这users个线程就会只针对同一个TestUserDAO实例进行测试。这个时候,我们在TestUserDAO里面就不能存放任何状态数据了。比如以前我很喜欢这么做:

    public class TestUserDAO extends TestCase{
     private int lastId;
     protected void setUp() throws Exception{
      super.setUp();
      //构造一个测试用的数据
      User user = new User("张三");
      //向数据库插入一条记录
      userDAO.addUser(user);
      lastId = user.getId();//刚刚插入的记录在数据库中产生的ID;
     }

     public void testUpdateUser() throws DaoException{
      //针对setup中插入的数据进行update操作
      user = userDAO.getUserById(lastId);
      user.setName("李四");
      userDAO.updateUser(user);
     }
     protected void tearDown() throws Exception{
      //删除测试数据
      userDAO.deleteUserById(lastId);
      super.tearDown();
     }
    这样我是通过一个int变量lastId在各个方法之间传递被测试的数据的。如果用junitperf来测试,lastId变量就会被后来的线程覆盖,导致测试失败。

    解决方法

    在com.clarkware.junitperf.TestFactory的文档中对这个问题做了说明:

    This factory class should be used in cases when a stateful test is intended to be decorated by a <code>LoadTest</code>.  A stateful test is defined as any test that defines test-specific state in its <code>setUp()</code> method.

    TestFactory的使用方法是这样:

    import com.clarkware.junitperf.TestFactory;

    ......

         suite.addTest(new LoadTest(new TestFactory(TestUserDAO.class), users,testTimes)); //并发负载测试

    但是这样只能观察整个测试类的表现。如果我们要单个的测试测试类中的一个测试,那么可以考虑另一种方法。我们在TestFactory的文档中看到:

     This class is dependent on Java 2.  For earlier platforms a  local cache implementation should be changed to use, for example,  a HashMap to track thread-local information.

    这个方法同事也适用于我们需要处理的情况:

    public class TestUserDAO extends TestCase{
     private static final ThreadLocal threadLocal = new ThreadLocal();
     protected void setUp() throws Exception{
      super.setUp();
      //构造一个测试用的数据
      User user = new User("张三");
      //向数据库插入一条记录
      userDAO.addUser(user);
      //lastId = user.getId();//刚刚插入的记录在数据库中产生的ID;
      threadLocal.set(new Integer(user.getId()));
     }

     public void testUpdateUser() throws DaoException{
      //针对setup中插入的数据进行update操作
      //user = userDAO.getUserById(lastId);
      user = userDAO.getUserById(((Integer)threadLocal.get()).intValue());
      user.setName("李四");
      userDAO.updateUser(user);
     }
     protected void tearDown() throws Exception{
      //删除测试数据
      userDAO.deleteUserById(((Integer)threadLocal.get()).intValue());
      super.tearDown();
     }

    author: emu(黄希彤)
  • 相关阅读:
    【小梅哥SOPC学习笔记】Altera SOPC嵌入式系统设计教程
    modelsim使用常见问题及解决办法集锦③
    modelsim使用常见问题及解决办法集锦 ②
    KeepAlived双主模式高可用集群
    充分利用nginx的reload功能平滑的上架和更新业务
    nginx日志配置指令详解
    MongoDB 副本集
    MongoDB 备份还原
    MongoDB的搭建、参数
    mongoDB整个文件夹拷贝备份还原的坑
  • 原文地址:https://www.cnblogs.com/stonehuang/p/6603242.html
Copyright © 2020-2023  润新知