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已经把最简单常用的单元测试功能包装成了一些带参数宏,因此在简单的测试中常常可以忽略命名空间的存在。
ASSERT宏
|
EXPECT宏
|
功能
|
ASSERT_TRUE
|
EXPECT_TRUE
|
判真
|
ASSERT_FALSE
|
EXPECT_FALSE
|
判假
|
ASSERT_EQ
|
EXPECT_EQ
|
相等
|
ASSERT_NE
|
EXPECT_NE
|
不等
|
ASSERT_GT
|
EXPECT_GT
|
大于
|
ASSERT_LT
|
EXPECT_LT
|
小于
|
ASSERT_GE
|
EXPECT_GE
|
大于或等于
|
ASSERT_LE
|
EXPECT_LE
|
小于或等于
|
ASSERT_FLOAT_EQ
|
EXPECT_FLOAT_EQ
|
单精度浮点值相等
|
ASSERT_DOUBLE_EQ
|
EXPECT_DOUBLE_EQ
|
双精度浮点值相等
|
ASSERT_NEAR
|
EXPECT_NEAR
|
浮点值接近(第3个参数为误差阈值)
|
ASSERT_STREQ
|
EXPECT_STREQ
|
C字符串相等
|
ASSERT_STRNE
|
EXPECT_STRNE
|
C字符串不等
|
ASSERT_STRCASEEQ
|
EXPECT_STRCASEEQ
|
C字符串相等(忽略大小写)
|
ASSERT_STRCASENE
|
EXPECT_STRCASENE
|
C字符串不等(忽略大小写)
|
ASSERT_PRED1
|
EXPECT_PRED1
|
自定义谓词函数,
(pred, arg1)(还有_PRED2, ...,
_PRED5)
|
- // add.h
- #pragma once
- inline int Add(int i, int j) { return i+j; }
- // add_unittest.cpp
- #include "add.h"
- #include
- 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
- #include
- 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);
ASSERT宏
|
EXPECT宏
|
功能
|
ASSERT_NO_THROW
|
EXPECT_NO_THROW
|
不抛出异常,参数为
(statement)
|
ASSERT_ANY_THROW
|
EXPECT_ANY_THROW
|
抛出异常,参数为
(statement)
|
ASSERT_THROW
|
EXPECT_THROW
|
抛出特定类型的异常,参数为
(statement, type)
|
- // divide.h
- #pragma once
- #include
- 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
- #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
参数值序列生成函数
|
含义
|
Bool()
|
生成序列
{false, true}
|
Range(begin, end[, step])
|
生成序列
{begin, begin+step, begin+2*step,
...} (不含
end),
step默认为1
|
Values(v1, v2,
...,
vN)
|
生成序列
{v1, v2,
...,
vN}
|
ValuesIn(container),
ValuesIn(iter1, iter2)
|
枚举STL
container,或枚举迭代器范围
[iter1, iter2)
|
Combine(g1, g2,
...,
gN)
|
生成
g1,
g2, ...,
gN的笛卡尔积,其中g1,
g2, ...,
gN均为参数值序列生成函数(要求C++0x的 |
- // 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;
- }
- // addupto_test.cpp
- #include
- #include "addupto.h"
- class AddUpToTest : public testing::TestWithParam
- {
- public:
- AddUpToTest() { n_ = GetParam(); }
- protected:
- unsigned n_;
- };
- TEST_P(AddUpToTest, Calibration) {
- EXPECT_EQ(NaiveAddUpTo(n_), FastAddUpTo(n_));
- }
- INSTANTIATE_TEST_CASE_P(
- NaiveAndFast, // prefix
- AddUpToTest, // test case name
- testing::Range(1u, 1000u) // parameters
- );