Ubuntu搭建gtest环境
在terminal中依次输入如下命令
# 下载googletest
$ sudo apt-get install libgtest-dev
# 如果没有安装cmake,执行该步骤安装
$ sudo apt-get install cmake
# 进入googletest目录
$ cd /usr/src/googletest
# 执行cmake && make
$ sudo cmake ./CMakeLists.txt
$ sudo make
$ sudo make install
测试代码
#include <gtest/gtest.h>
TEST(GTest, HandleGTest)
{
std::cout << "This is a test!" << std::endl;
}
int main(int argc,char*argv[])
{
testing::InitGoogleTest(&argc,argv);
return RUN_ALL_TESTS();
}
编译命令:
g++ gtest.cpp -lgtest -lpthread
结果:
语法规则
测试代码运行顺序
从上到下顺序执行每个TEST_F或者TEST宏
TEST_F 和 TEST
TEST_F提供了一个初始化函数(SetUp)和一个清理函数(TearDown),在TEST_F中使用的变量可以在初始化函数SetUp中初始化,在TearDown中销毁,并且所有的TEST_F是互相独立的,都是在初始化以后的状态开始运行,一个TEST_F不会影响另一个TEST_F所使用的数据。
/*
TEST语法定义:
-test_case_name第一个参数是测试用例名, 通常是取测试函数名或者测试类名
-test_name 第二个参数是测试名这个随便取
-当测试完成后显示的测试结果将以"测试用例名.测试名"的形式给出
*/
TEST(test_case_name, test_name);
/*
TEST_F语法定义:
-test_case_name第一个参数是测试用例名,必须取类名
-test_name 第二个参数是测试名这个随便取
-使用TEST_F时必须继承::testing::Test类。并且该类提供了两个接口void SetUp(); void TearDown();
*/
TEST_F(test_case_name, test_name);
TEST_F 示例
#include <gtest/gtest.h>
class BaseAPI {
public:
virtual int getValue() = 0;
virtual void setValue(int) = 0;
};
class Base : public BaseAPI {
public:
Base(int value) {
std::cout << "Create Base" << std::endl;
this->m_value = value;
}
~Base() {
std::cout << "Destory Base" << std::endl;
}
int getValue() {
return this->m_value;
}
void setValue(int value) {
this->m_value = value;
}
private:
int m_value;
};
class BaseTest : public ::testing::Test {
protected:
void SetUp() override {
std::cout << "SetUp BaseTest" << std::endl;
m_base = std::make_shared<Base>(999);
}
void TearDown() override {
std::cout << "SetUp TearDown" << std::endl;
}
std::shared_ptr<Base> m_base;
};
TEST_F(BaseTest, testValue)
{
std::cout << "[1] value: " << m_base->getValue() << std::endl;
m_base->setValue(777);
std::cout << "[2] value: " << m_base->getValue() << std::endl;
}
int main(int argc,char*argv[])
{
testing::InitGoogleTest(&argc,argv);
return RUN_ALL_TESTS();
}
运行结果
模拟类Mock及其常用宏
一般正常类的类名前面加上Mock前缀,表示模拟类。模拟类需要继承正常类。
MOCK_METHODx
x可以是0、1、2、3、4…,模拟的函数有多少个参数x就等于多少。
MOCK_CONST_METHODx
类似于MOCK_METHODx,只不过被模拟的函数是一个常量成员函数。
示例
class BaseAPI {
public:
virtual std::string getName() = 0;
virtual void setName(std::string name) = 0;
virtual void printName() const = 0;
};
class MockBaseAPI : public BaseAPI {
public:
MOCK_METHOD1(setName, void(std::string name));
MOCK_METHOD0(getName, std::string());
MOCK_CONST_METHOD0(printName, void());
};
EXPECT_CALL
使用MOCK_METHODx宏声明的类成员函数,使用EXPECT_CALL等一系列宏来指定这些函数在被调用时所做的动作。
/*
-mock_object: Mock对象
-method(matcher): method是Mock类中的函数,matcher匹配器实例,可以为_,表示不关心参数,参数为任何东西
-Times(cardinality): 函数应该被调用的次数是cardinality次,调用需要在同一个TEST_F
-WillOnce(action):执行一次method,所期望的action
-WillRepeatedly(action):每次执行method,所期望的action
-action: 希望应该怎么办,如果不设定预期action,则使用默认action
-如果使用WillOnce,那么WillOnce的数量需要和Times设定的次数匹配,或者直接不指定Times
-如果使用WillRepeatedly,那么就不关心跟Times匹配的问题
*/
EXPECT_CALL(mock_object, method(matcher))
.Times(cardinality)
.WillOnce(action)
.WillRepeatedly(action)
示例
#include <gtest/gtest.h>
#include <gmock/gmock.h>
using ::testing::_;
using ::testing::Return;
class BaseAPI {
public:
virtual int getValue() = 0;
virtual void setValue(int value) = 0;
virtual void printValue(int value) const = 0;
};
class MockBaseAPI : public BaseAPI {
public:
MOCK_METHOD0(getValue, int());
MOCK_METHOD1(setValue, void(int value));
MOCK_CONST_METHOD1(printValue, void(int value));
};
class BaseTest : public ::testing::Test {
protected:
void SetUp() override {
std::cout << "SetUp BaseTest" << std::endl;
}
void TearDown() override {
std::cout << "SetUp TearDown" << std::endl;
}
};
TEST_F(BaseTest, testValue)
{
MockBaseAPI* m_MockBaseAPI = new MockBaseAPI();
// 期望后续会调用getValue() 3次,期望调用的返回值分别是3/6/9。如果getValue()少调用一次,或者多调用一次,都会出现错误。
// 有3个WillOnce,Times必须要等于3
EXPECT_CALL(*m_MockBaseAPI, getValue()).Times(3)
.WillOnce(Return(3))
.WillOnce(Return(6))
.WillOnce(Return(9));
/*
EXPECT_CALL(*m_MockBaseAPI, getValue()).WillRepeatedly(Return(9));
// 这种方式,getValue()函数可以被调用无数次,每次都是返回9,也可以一次都不调用,不会报错
*/
std::cout << m_MockBaseAPI->getValue() << std::endl; // 打印 3
std::cout << m_MockBaseAPI->getValue() << std::endl; // 打印 6
std::cout << m_MockBaseAPI->getValue() << std::endl; // 打印 9
// 期望后续会调用setValue() 1次,不关心参数
EXPECT_CALL(*m_MockBaseAPI, setValue(_)).Times(1);
m_MockBaseAPI->setValue(1); // 因为不关心参数,所以传入任意数值都会成功
// 期望后续会调用printValue() 1次,参数必须是3,如果不是3就是fail
EXPECT_CALL(*m_MockBaseAPI, printValue(3)).Times(1);
m_MockBaseAPI->printValue(3);
delete m_MockBaseAPI;
}
int main(int argc,char*argv[])
{
testing::InitGoogleTest(&argc,argv);
return RUN_ALL_TESTS();
}
正常的调用结果
少了一次getValue调用的结果
printValue传入的参数不是3的结果
ASSERT 和 EXPECT 断言宏
ASSERT 是如果当前点检测失败则退出当前函数,EXPECT 是如果当前点检测失败则继续往下执行,宏类型类似ASSERT。
BOOL类型
-
ASSERT_TRUE (参数)
:期望值为 参数 = TRUE -
ASSERT_FALSE (参数)
:期望值为 参数 = FALSE
数值类型
-
ASSERT_EQ (参数1, 参数2)
:期望值为 参数1 == 参数2 -
ASSERT_NE (参数1, 参数2)
:期望值为 参数1 != 参数2 -
ASSERT_LT (参数1, 参数2)
:期望值为 参数1 < 参数2 -
ASSERT_GT (参数1, 参数2)
:期望值为 参数1 > 参数2 -
ASSERT_LE (参数1, 参数2)
:期望值为 参数1 <= 参数2 -
ASSERT_GE (参数1, 参数2)
:期望值为 参数1 >= 参数2
字符串类型
字符串的类型为C语言类型的字符串
-
ASSERT_STREQ (字符串1, 字符串2)
:期望值为 字符串1 == 字符串2 -
ASSERT_STRNE (字符串1, 字符串2)
:期望值为 字符串1 != 字符串2
示例
#include <gtest/gtest.h>
#include <gmock/gmock.h>
using ::testing::_;
using ::testing::Return;
class BaseAPI {
public:
virtual int getValue() = 0;
virtual void setValue(int value) = 0;
virtual void printValue(int value) const = 0;
};
class MockBaseAPI : public BaseAPI {
public:
MOCK_METHOD0(getValue, int());
MOCK_METHOD1(setValue, void(int value));
MOCK_CONST_METHOD1(printValue, void(int value));
};
class BaseTest : public ::testing::Test {
protected:
void SetUp() override {
std::cout << "SetUp BaseTest" << std::endl;
}
void TearDown() override {
std::cout << "SetUp TearDown" << std::endl;
}
};
TEST_F(BaseTest, testValue)
{
MockBaseAPI* m_MockBaseAPI = new MockBaseAPI();
// 期望后续会调用getValue() 3次,期望每次调用的返回值都是3。如果getValue()少调用一次,或者多调用一次,都会出现错误
EXPECT_CALL(*m_MockBaseAPI, getValue()).Times(3).WillRepeatedly(Return(3));
// 判断getValue()应该返回的值是否等于30,判断失败打印error log: [EXPECT_EQ] getValue FAIL!,但是下面的步骤会继续执行
EXPECT_EQ(30, m_MockBaseAPI->getValue()) << "[EXPECT_EQ] getValue FAIL!";
// 上面判断失败了,但是会继续执行。判断getValue()应该返回的值是否等于30,判断失败打印error log: [ASSERT_EQ 1] getValue FAIL!,同时下面的步骤会被中断
ASSERT_EQ(30, m_MockBaseAPI->getValue()) << "[ASSERT_EQ 1] getValue FAIL!";
// 上面的ASSERT_EQ执行失败,下面这句不会被执行到
ASSERT_EQ(30, m_MockBaseAPI->getValue()) << "[ASSERT_EQ 2] getValue FAIL!";
delete m_MockBaseAPI;
}
int main(int argc,char*argv[])
{
testing::InitGoogleTest(&argc,argv);
return RUN_ALL_TESTS();
}
测试结果
匹配器 MATCHER_Px
x的值可以是空、2、3…,表示的是匹配器的参数个数,匹配器名字不是参数。MATCHER_P(name, parameter, ""):一个参数
MATCHER_P2(name, parameter1, parameter2, ""):两个参数
MATCHER_P3(name, parameter1, parameter2, parameter3, ""):三个参数
/*
-NameMatcher: 匹配器的名字,调用的时候调用该名字
-str: 调用匹配器的时候,传入的参数
-arg: 调用使用该匹配器的函数的时候,传入的参数
*/
MATCHER_P(NameMatcher, str, "") {
return arg == str ? true : false;
}
示例
#include <gtest/gtest.h>
#include <gmock/gmock.h>
using ::testing::_;
using ::testing::Return;
// 函数调用的时候,函数的参数必须是等于该value
MATCHER_P(NameMatcher, value, "") {
return arg == value ? true : false;
}
// 传入的参数value必须是369,否则直接报错。后续函数调用的时候,函数的参数必须是等于该value(369)
MATCHER_P(NameMatcher2, value, "") {
if (value != 369) return false;
return arg == value ? true : false;
}
class BaseAPI {
public:
virtual int getValue() = 0;
virtual void setValue(int value) = 0;
virtual void printValue(int value) const = 0;
};
class MockBaseAPI : public BaseAPI {
public:
MOCK_METHOD0(getValue, int());
MOCK_METHOD1(setValue, void(int value));
MOCK_CONST_METHOD1(printValue, void(int value));
};
class BaseTest : public ::testing::Test {
protected:
void SetUp() override {
std::cout << "SetUp BaseTest" << std::endl;
}
void TearDown() override {
std::cout << "SetUp TearDown" << std::endl;
}
};
TEST_F(BaseTest, testValue)
{
MockBaseAPI* m_MockBaseAPI = new MockBaseAPI();
// 期望后续会调用setValue()1次,对传入的参数不在意
EXPECT_CALL(*m_MockBaseAPI, setValue(_)).Times(1);
m_MockBaseAPI->setValue(123456); // 对参数不在意,所以可以是任意值
// 期望后续会调用setValue()1次,传入的参数必须要是3,不是3就会失败
EXPECT_CALL(*m_MockBaseAPI, setValue(NameMatcher(3))).Times(1);
m_MockBaseAPI->setValue(3);
// 期望后续会调用setValue()1次,NameMatcher2中传入的参数必须是369否则直接报错,另外调用的时候必须传入369,否则也会失败
EXPECT_CALL(*m_MockBaseAPI, setValue(NameMatcher2(369))).Times(1);
m_MockBaseAPI->setValue(369);
delete m_MockBaseAPI;
}
int main(int argc,char*argv[])
{
testing::InitGoogleTest(&argc,argv);
return RUN_ALL_TESTS();
}
NameMatcher(3) 但是函数执行传入的不是3
NameMatcher2(369) 匹配器传入的参数不是369
NameMatcher2(369) 匹配器传入的参数是369,但是函数执行传入的不是369
gtest_filter过滤器规则
--gtest_filter=testClass.testCase
:仅运行testClass下的testCase--gtest_filter=*str*
:仅运行包含str字符串的测试case--gtest_filter=-*str*
:包含字符串str的测试case跳过运行--gtest_filter=-*str1*:*str2*
:包含"str1"或"str2"的测试case跳过运行--gtest_filter=*str*:-*str1*:*str2*
:这将运行包含str且不包含str1或str2的测试case“-”
后的任何内容都将计入排除列表。