深入理解gtest C/C++单元测试经验谈
Google C++ Testing Framework(简称gtest,http://code.google.com/p/googletest/)是Google公司发布的一个开源C/C++单元测试框架,已被应用于多个开源项目及Google内部项目中,知名的例子包括Chrome Web浏览器、LLVM编译器架构、Protocol Buffers数据交换格式及工具等。
优秀的C/C++单元测试框架并不算少,相比之下gtest仍具有明显优势。与CppUnit比,gtest需要使用的头文件和函数宏更集中,并支持测试用例的自动注册。与CxxUnit比,gtest不要求Python等外部工具的存在。与Boost.Test比,gtest更简洁容易上手,实用性也并不逊色。Wikipedia给出了各种编程语言的单元测试框架列表(http://en.wikipedia.org/wiki/List_of_unit_testing_frameworks)。
一、基本用法
gtest当前的版本是1.5.0,如果使用Visual C++编译,要求编译器版本不低于7.1(Visual C++ 2003)。如下图所示,它的msvc文件夹包含Visual C++工程和项目文件,samples文件夹包含10个使用范例。

一般情况下,我们的单元测试代码只需要包含头文件gtest.h。gtest中常用的所有结构体、类、函数、常量等,都通过命名空间testing访问,不过gtest已经把最简单常用的单元测试功能包装成了一些带参数宏,因此在简单的测试中常常可以忽略命名空间的存在。
// add.h
#pragma once
inline int Add(int i, int j) { return i+j; } // add_unittest.cpp
#include "add.h"
#include <gtest/gtest.h>
TEST(Add, 负数) {
EXPECT_EQ(Add(-1,-2), -3);
EXPECT_GT(Add(-4,-5), -6); // 故意的
}
TEST(Add, 正数) {
EXPECT_EQ(Add(1,2), 3);
EXPECT_GT(Add(4,5), 6);
} ASSERT_EQ(M[i], N[j]) << "i = " << i << ", j = " << j;
// gtest-main.cc
int main(int argc, char **argv) {
std::cout << "Running main() from gtest_main.cc\n";
testing::InitGoogleTest(&argc, argv);
return RUN_ALL_TESTS();
} 
// add_unittest2.cpp
#include "add.h"
#include <stdio.h>
#include <gtest/gtest.h>
class AddTest: public testing::Test
{
public:
virtual void SetUp() { puts("SetUp()"); }
virtual void TearDown() { puts("TearDown()"); }
};
TEST_F(AddTest, 正数) {
ASSERT_GT(Add(1,2), 3); // 故意的
ASSERT_EQ(Add(4,5), 6); // 也是故意的
} class Environment {
public:
virtual ~Environment() {}
virtual void SetUp() {}
virtual void TearDown() {}
}; Environment* AddGlobalTestEnvironment(Environment* env);
// divide.h
#pragma once
#include <stdexcept>
int divide(int dividend, int divisor) {
if(!divisor) {
throw std::length_error("can't be divided by 0"); // 故意的
}
return dividend / divisor;
} // divide-unittest.cpp
#include <gtest/gtest.h>
#include "./divide.h"
TEST(Divide, ByZero) {
EXPECT_NO_THROW(divide(-1, 2));
EXPECT_ANY_THROW({
int k = 0;
divide(k, k);
});
EXPECT_THROW(divide(100000, 0), std::invalid_argument);
} 
try {
statement;
}
catch(type const&) {
// throw
}
catch(...) {
// any throw
}
// no throw // addupto.h
#pragma once
inline unsigned NaiveAddUpTo(unsigned n) {
unsigned sum = 0;
for(unsigned i = 1; i <= n; ++i) sum += i;
return sum;
}
inline unsigned FastAddUpTo(unsigned n) {
return n*(n+1)/2;
}