`
chelsea
  • 浏览: 119330 次
  • 来自: ...
社区版块
存档分类
最新评论

设计原则与模式: 案例介绍--CppUnit

    博客分类:
 
阅读更多

设计原则与模式: 案例介绍--CppUnit

CppUnit 是一个单元测试框架, 我们看一看它的设计是如何遵循基础的设计原则和模式的

单一职责原则

TestRunner 和 TestResult 的分离

class CPPUNIT_API TestRunner {

virtual void addTest( Test *test );

virtual void run( TestResult &result, const std::string &testPath = "" );

...

};

TestRunner 负责收集并运行测试用例, 但并不主动打印测试结果. 测试结果被收集在 TestResult 对象中, 可以以各种形式被处理:

CPPUNIT_NS::TestResult controller;

CPPUNIT_NS::TestResultCollector result;

controller.addListener( &result );

...

runner.run( controller );

// Print test in a compiler compatible format.

CPPUNIT_NS::CompilerOutputter compiler_outputter( &result, CPPUNIT_NS::stdCOut() );

compiler_outputter.write();

// Print test in XML format.

CPPUNIT_NS::XmlOutputter xml_outputter( &result, CPPUNIT_NS::stdCOut() );

xml_outputter.write();

 

 

开放封闭原则

遵循开放封闭原则的一个重要特征就是 "针对接口/基类编程", 任何根据 typeid 等类型信息进行的分支处理如 if/else, switch/case 等都可以看做是破坏开放封闭原则的前兆

TestResult 对于如何处理测试过程中发生的事件是开放的, 可以通过 TestListener 来扩展

class CPPUNIT_API TestResult : protected SynchronizedObject {

virtual void addListener( TestListener *listener );

virtual void removeListener( TestListener *listener );

/// Informs TestListener that a test will be started.

virtual void startTest( Test *test );

...

};

 


void TestResult::startTest( Test *test ) {

...

for ( TestListeners::iterator it = m_listeners.begin(); it != m_listeners.end(); ++it )

(*it)->startTest( test );

}

};

 

 

void BriefTestProgressListener::startTest ( Test *test ) {

stdCOut () << test-> getName ();

stdCOut ().flush();

m_lastTestFailed = false ;

}

void TextTestProgressListener::startTest ( Test *test ){

stdCOut() << "." ;

}

接口分离原则

从 TestRunner 的角度来讲, 它只需要把测试用例 "运行" 起来, 并不需要关心测试用例是如何运行的, 如何搭建和清理运行环境, 它所需要的 Test 对象的接口, 只需要一个 run( ) 方法 :

class CPPUNIT_API Test {

virtual void run ( TestResult *result ) =0;

};

void TestRunner::addTest ( Test *test ) {

m_suite -> addTest ( test );

}

void TestRunner::run ( TestResult &controller, const std:: string &testPath ) {

TestPath path = m_suite -> resolveTestPath ( testPath );

Test *testToRun = path.getChildTest();

controller. runTest ( testToRun );

}

但是总有一个类需要负责搭建和清理测试环境:

class CPPUNIT_API TestFixture {

virtual void setUp () {};

virtual void tearDown () {};

};

class CPPUNIT_API TestCase : public Test , public TestFixture {

};

TestCase 这个类分别实现了TestTestFixture 两个分离的接口, 这样TestCase 的客户只需要按需依赖 Test TestFixture 就可以了

 

 

依赖倒置原则

后来被发展为依赖注入或者控制反转模式,思想就是依赖于抽象,而不是具体的类

TestRunner 并不依赖具体的 TestCase/TestSuite 等, 它只依赖于 Test. 在main函数中由 TestFactoryRegistry::makeTest() 产生 具体的Test对象, 注入到TestRunner中

class CPPUNIT_API TestRunner {

virtual void addTest( Test *test );

virtual void run( TestResult &result, const std::string &testPath = "" );

...

};

int main ( int argc, char * argv[] ) {

// Add the top suite to the test runner

CPPUNIT_NS:: TestRunner runner;

runner. addTest ( CPPUNIT_NS:: TestFactoryRegistry :: getRegistry (). makeTest () );

runner. run ( controller );

...

}


Test * TestFactoryRegistry::makeTest () {

TestSuite *suite = new TestSuite ( m_name );

addTestToSuite ( suite );

return suite;

}

 

Composite 模式

TestSuite 本身也是 Test, 但可以包含很多其它 Test

class CPPUNIT_API TestComposite : public Test {

...

};

class CPPUNIT_API TestSuite : public TestComposite {

void addTest ( Test *test );

};

客户代码参见前面的 main 函数

 

Decorator 模式

1. 如何重复运行测试用例?

重复运行也是运行, 只是添加了额外的一种职责: 运行多次

class CPPUNIT_API TestDecorator : public Test {

...

};

class CPPUNIT_API RepeatedTest : public TestDecorator {

RepeatedTest ( Test *test, int timesRepeat ) : TestDecorator ( test ), m_timesRepeat (timesRepeat) {

}

void run ( TestResult *result ) {

for ( int n = 0; n < m_timesRepeat; n++ ){

TestDecorator::run( result );

}

}

...

};

 

2. 如何确保 TestCase 的隔离性, 即使出现异常也不影响后续 TestCase 的运行 ?

ProtectedFunctor -> Functor -> 函数指针

class TestCaseMethodFunctor : public Functor {

typedef void ( TestCase ::* Method )();

TestCaseMethodFunctor ( TestCase *target, Method method ) : m_target ( target ), m_method ( method ) {

}

bool operator() () const {

( m_target ->* m_method )();

return true ;

}

...

};

bool DefaultProtector::protect ( const Functor &functor, const ProtectorContext &context ) {

try {

return functor();

} catch ( Exception &failure ) {

reportFailure( context, failure );

} catch ( std::exception &e ) {

reportError( context, Message( "uncaught exception of type " , e.what() ) );

} catch ( ... ) {

reportError( context, Message( "uncaught exception of unknown type" ) );

}

return false ;

}

练习 : 这里使用了 Decorator 模式的思想, 但并不是严格的 Decorator 模式, 为什么?

 

Observer 模式

TestResult 与 TestListener

void TestResult::addListener ( TestListener *listener ) {

m_listeners .push_back( listener );

}

void TestResult::removeListener ( TestListener *listener ) {

removeFromSequence( m_listeners , listener );

}

void TestResult::startTestRun ( Test *test ) {

for ( TestListeners :: iterator it = m_listeners .begin(); it != m_listeners .end(); ++it )

(*it)->startTestRun( test, this );

}

 

void TestResult::endTestRun ( Test *test ) {

for ( TestListeners :: iterator it = m_listeners .begin(); it != m_listeners .end(); ++it )

(*it)->endTestRun( test, this );

}

 

Factory Method 模式

调用测试用例注册表来产生测试用例

class CPPUNIT_API TestFactoryRegistry : public TestFactory {

virtual Test *makeTest ();

};

Strategy 模式

不同的格式的输出, 如Compiler风格的, Xml格式的, 等等

class CPPUNIT_API TextTestRunner : public CPPUNIT_NS:: TestRunner {

void setOutputter ( Outputter *outputter ) {
delete m_outputter ;
m_outputter = outputter;
}

void printResult ( bool doPrintResult ) {

m_outputter -> write ();

}

};


CPPUNIT_NS:: TextTestRunner runner;

CPPUNIT_NS:: CompilerOutputter compiler_outputter ( ... );

runner.setOutputter(compiler_outputter);

Template Method 模式

TestFixture 的 setUp 与 tearDown

class CPPUNIT_API TestFixture {

virtual void setUp () {};

virtual void tearDown () {};

};

void TestCase::run ( TestResult *result ) {

result-> startTest ( this );

if ( result->protect( TestCaseMethodFunctor( this , & TestCase ::setUp ), this , "setUp() failed" ) ) {

result->protect( TestCaseMethodFunctor( this , & TestCase :: runTest ), this );

}

result->protect( TestCaseMethodFunctor( this , & TestCase ::tearDown ), this , "tearDown() failed" );

result-> endTest ( this );

}

 

练习 : CppUnit 支持将测试结果输出到控制台, 或者文件, 但缺省并不支持同时输出到控制台和文件, 如何在遵循各种设计原则的情况下, 为 CppUnit 添加此功能?

分享到:
评论

相关推荐

    编译好的cppunit-1.12.1 与ProjectWizard

    **编译好的cppunit-1.12.1与ProjectWizard**是针对C++开发者的重要工具,它们在软件开发过程中扮演着关键角色。cppunit-1.12.1是著名的C++单元测试框架,而ProjectWizard则是一个帮助创建新项目的工作流程辅助工具。...

    cppunit-1.12.0 单元测试工具

    ### 安装与使用cppunit-1.12.0 在cppunit-1.12.0压缩包中,通常包含了源代码、文档、示例等资源。安装过程包括编译源代码并将其链接到项目中。开发者可以通过以下步骤开始使用cppunit: 1. 解压cppunit-1.12.0...

    cppunit-1.10.2.tar.gz

    `cppunit-1.10.2.tar.gz` 是一个包含 CppUnit 1.10.2 版本源代码的压缩包。CppUnit 是一个用于 C++ 语言的单元测试框架,它模仿了 Java 的 JUnit 框架,并在 C++ 开发环境中提供了一种组织和执行单元测试的工具。 #...

    cppunit-1.12.0

    cppunit-1.12.0 是一个针对C++编程语言的开源单元测试框架,它在软件开发过程中扮演着重要的角色。单元测试是软件工程中的关键实践,允许开发者独立地验证代码的各个小部分,确保它们按预期工作。cppunit 提供了丰富...

    cppunit-1.12.1.zip

    "cppunit-1.12.1.zip" 这个标题表明我们正在讨论一个关于CppUnit的特定版本,即1.12.1。CppUnit是一个开源的单元测试框架,用于C++编程语言。这个压缩包很可能包含了CppUnit库的所有源代码、头文件、库文件以及可能...

    cppunit-1.12.1 用于单元测试的工具

    `cppunit-1.12.1` 是一个广泛使用的开源单元测试框架,它专门为C++编程语言设计,帮助开发者编写和运行针对其代码的自动化测试。单元测试是软件开发过程中的一个重要环节,通过编写独立的小型测试用例来验证程序的...

    cppunit的使用

    - 在“系统属性”的“环境变量”中添加 `E:\cppunit-1.12.0\include` 和 `E:\cppunit-1.12.0\lib` 到 Path 变量中。 3. **项目信息配置**: - 在项目的“设置”对话框中,选择“链接器”选项卡,填写库文件名如 `...

    CPPUNIT单元测试 -- 测试程序模板

    CPPUNIT,全称为C++ Unit,是用于C++编程语言的一个单元测试框架,它借鉴了JUnit在Java中的设计理念。单元测试是一种软件开发的最佳实践,它允许开发者对代码的各个独立部分进行验证,确保它们按预期工作。CPPUNIT的...

    subunit-cppunit-devel-1.4.0-1.el8.aarch64.rpm

    官方离线安装包,亲测可用

    cppunit-1.12.1

    `cppunit-1.12.1` 是 `cppunit` 库的一个特定版本,它是一个针对C++编程语言设计的单元测试框架。单元测试是软件开发过程中的一个重要组成部分,通过编写小规模的独立测试用例来验证代码模块的功能正确性。`cppunit`...

    VS2008_CppUnit

    - 首先,你需要下载CppUnit的源代码,例如这里提到的cppunit-1.10.2版本。 - 使用VS2008打开源代码并编译项目,生成所需的库文件(.lib)和头文件(.h)。 - 将编译后的库文件添加到你的系统路径或项目的链接器...

    cppunit

    文档`CppUnit源码解读.doc`和`CPPUnit--源码导读.doc`可能会深入解析cppunit的内部实现,包括类结构、主要接口以及关键算法。这些文档可以帮助你理解cppunit如何管理测试用例、如何调度和执行测试,以及如何处理测试...

    cppunit在linux下的安装配置及测试

    - 需要手动将 `cppunit-1.10.2/cppunit` 目录复制到 `/usr/include`,以便在其他项目中可以引用这些头文件。 4. **环境变量设置**: - 在运行包含cppunit的程序时,需要设置 `LD_LIBRARY_PATH` 环境变量,指向...

    Cppunit-1.12.1以及测试框架模板实例Demo

    `Cppunit-1.12.1` 是一个开源的单元测试框架,用于C++编程语言。它是基于Java的JUnit测试框架设计的,为C++开发者提供了一种结构化和自动化测试代码的方式。在C++的世界里,单元测试是确保代码质量、预防错误和简化...

    单元测试培训时的cppunit相关的资料

    2007-03-26 20:08 446,921 cppunit-docs-1.12.0.tar.gz 2007-03-26 20:09 177,009 MSDNIntegrator-1.0.zip 4 个文件 1,610,335 字节 2 个目录 422,510,592 可用字节 G:\培训资料\cppunit\Install 的目录 2008-...

Global site tag (gtag.js) - Google Analytics