简单易懂的单元测试框架-gtest

Posted bluepcghost

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了简单易懂的单元测试框架-gtest相关的知识,希望对你有一定的参考价值。


简介

    事件机制用于在案例运行前后添加一些操作(相当于挂钩函数)。目前,gtest提供了三种等级的事件,分别:

  • 全局级,所有案例执行的前后
  • TestSuite级,某一个案例集的前后
  • TestCase级,每一个案例的前后

    不同等级的事件在影响不同的范围。


示例-全局级

    全局级事件在所有案例执行的前后运行,制作该事件需要继承testing::Environment类,实现里面的SetUp和TearDown方法。为了模拟实际应用中的情况,下面的测试代码分为了三个文件,分别代表:全局级事件头文件(unit_test.h);全局级事件实现文件(unit_test.cc);具体的测试文件(test_main.cc)。测试代码如下,

/* 为单元测试设置全局事件 */
// unit_test.h
#ifndef UNIT_TEST_H_
#define UNIT_TEST_H_

#include <gtest/gtest.h>

class FooEnvironment: public testing::Environment{
  public:
    void SetUp();
    void TearDown();
};

#endif
/* 为单元测试设置全局事件 */
// unit_test.cc
#include <iostream>
#include "unit_test.h"

void FooEnvironment::SetUp(){
  std::cout<<"FooEnvironment::SetUp"<<std::endl;
}

void FooEnvironment::TearDown(){
  std::cout<<"FooEnvironment::TearDown"<<std::endl;
}
/* 单元测试 */
// test_main.cc
#include "unit_test.h"

bool checkOdd(int a){
  return a%2==1;
}
// 测试集checkOddTest的测试1
TEST(checkOddTest, test1){
  ASSERT_EQ(true, checkOdd(1));
  ASSERT_EQ(true, checkOdd(11));
}

int main(int argc, char **argv){
  testing::AddGlobalTestEnvironment(new FooEnvironment);
  testing::InitGoogleTest(&argc, argv);
  return RUN_ALL_TESTS();
}

    上述代码是一个简单的全局级事件模板,大部分时候我们都可以采用类似的结构在实际工程中添加单元测试程序。其中,SetUp方法在所有案例前执行,TearDown方法在所有案例后执行。
    这份代码的编译同样需要gtest的头文件和库文件,所以采用了与简介中相同的方案来配置编译环境。cmake文件如下,

cmake_minimum_required(VERSION 2.6)

# Locate GTest
find_package(GTest REQUIRED)
include_directories(${GTEST_INCLUDE_DIRS})

add_executable(runTests test_main.cc
                        unit_test.cc)
target_link_libraries(runTests ${GTEST_LIBRARIES} pthread)

    运行如下编译指令,可以获得单元测试程序,

cmake CMakeLists.txt
make

    now,运行单元测试程序,结果如下,

[==========] Running 1 test from 1 test case.
[----------] Global test environment set-up.
FooEnvironment::SetUp
[----------] 1 test from checkOddTest
[ RUN ] checkOddTest.test1
[ OK ] checkOddTest.test1 (0 ms)
[----------] 1 test from checkOddTest (0 ms total)

[----------] Global test environment tear-down
FooEnvironment::TearDown
[==========] 1 test from 1 test case ran. (0 ms total)
[ PASSED ] 1 test.

    可以看到,我们的单元测试程序汇报了单元测试的最终结果,包括:全局级事件的运行、测试样例的运行等等。


示例-TestSuite级

    TestSuite级的事件覆盖范围相对要小一些,只是覆盖几个具备相同事件需求的测试样例,该事件需要继承自testing::Test,实现其中的两个静态方法,

  1. SetUpTestCase()方法在该测试集的第一个TestCase之前执行
  2. TearDownTestCase()方法在该测试集的最后一个TestCase之后执行

    下列示例代码采用了与全局级事件相似的组织结构。测试代码如下,

/* test_suit.h */
/* 测试TestSuite事件 */
#ifndef TEST_SUITE_H_
#define TEST_SUITE_H_

#include <gtest/gtest.h>

class FooTest: public testing::Test{
  protected:
    static void SetUpTestCase();
    static void TearDownTestCase();
    // some expensive resource shared by all tests.
    static int *p_;
};

#endif
/* test_suit.cc */
/* 测试TestSuite事件 */
#include <iostream>
#include "test_suit.h"

int* FooTest::p_ = nullptr;

void FooTest::SetUpTestCase(){
  p_ = new int;
  std::cout<<"FooTest::SetUpTestCase"<<std::endl;
}

void FooTest::TearDownTestCase(){
  delete p_;
  p_ = nullptr;
  std::cout<<"FooTest::SetUpTestCase"<<std::endl;
}
/* test_main.cc */
/* 测试TestSuite事件 */
#include "test_suit.h"

bool isFoo(int data){
  return data%2==1;
}

TEST_F(FooTest, test1){
  ASSERT_EQ(true, isFoo(3));
  ASSERT_EQ(false, isFoo(4));
  ASSERT_EQ(true, isFoo(2));
}

int main(int argc, char **argv){
  testing::InitGoogleTest(&argc, argv);
  return RUN_ALL_TESTS();
}

    使用的编译文件为,

cmake_minimum_required(VERSION 2.6)
set(CMAKE_CXX_STANDARD 11)

# Locate GTest
find_package(GTest REQUIRED)
include_directories(${GTEST_INCLUDE_DIRS})

add_executable(runTests test_main.cc
                        test_suit.cc)
target_link_libraries(runTests ${GTEST_LIBRARIES} pthread)

    使用如下指令编译,

cmake CMakeLists.txt
make

    当编译好单元测试程序后,运行该程序,结果如下,

[email protected]:~/lab/gtest/event/suite$ ./runTests
[==========] Running 1 test from 1 test case.
[----------] Global test environment set-up.
[----------] 1 test from FooTest
FooTest::SetUpTestCase
[ RUN ] FooTest.test1
/home/blue/lab/gtest/event/suite/test_main.cc:11: Failure
Value of: isFoo(2)
Actual: false
Expected: true
[ FAILED ] FooTest.test1 (0 ms)
FooTest::SetUpTestCase
[----------] 1 test from FooTest (0 ms total)

[----------] Global test environment tear-down
[==========] 1 test from 1 test case ran. (1 ms total)
[ PASSED ] 0 tests.
[ FAILED ] 1 test, listed below:
[ FAILED ] FooTest.test1

1 FAILED TEST

    单元测试运行结果的展示也几乎一样。


示例-TestCase级

    该部分与上述两部分非常相似,仅仅是继承类与重载函数稍有不同。


Reference

    [1] http://www.cnblogs.com/coderzh/archive/2009/04/06/1430396.html 本文部分参考了这里





























以上是关于简单易懂的单元测试框架-gtest的主要内容,如果未能解决你的问题,请参考以下文章

gtest使用小结

单元测试框架GoogleTest

Google开源C++单元测试框架Google Test

gtest从一无所知到熟练使用如何用gtest写单元测试

MFC程序使用GTest搭建测试框架

还在用gtest?更好用的测试框架介绍