玩转单元测试之gmock

引言

前文我们学习了gtest相关的使用,单靠gtest,有些场景仍然无法进行测试,因此就诞生了gmock。

gmock快速入门

在引入gtest时,gmock也同样引入了,因此只需要在编译时加上合适的编译选项即可,注意不同版本的gtest在使用上可能会有所不同,当前为v1.10版本。

1. 待测试文件

cpp 复制代码
#ifndef __HELLO_H__
#define __HELLO_H__

#include <iostream>
#include <string>

class Tigger{
public:
    Tigger()
    {

    }
    virtual ~Tigger() {}
    virtual bool eat(const std::string& food)
    {
        if (food == "meat") {
            return true;
        }
        return false;
    }
};
class FeedTigger {
public:
    Tigger* _tigger;
    FeedTigger(Tigger* tigger)
    {
        _tigger = tigger;
    }
    bool feed(const std::string& food)
    {
        if (_tigger !=NULL && _tigger->eat(food)) {
            return true;
        }
        return false;
    }
};
#endif

2. 单测文件

cpp 复制代码
#include "hello.h"
#include "gtest/gtest.h"
#include "gmock/gmock.h"

using namespace ::testing;

namespace {

class MockTigger : public Tigger {
public:
	MOCK_METHOD1(eat, bool(const std::string&));
};
TEST(TestTigger, CaseEat) 
{
    Tigger tigger;
    FeedTigger feed_tigger0(&tigger);
    EXPECT_FALSE(feed_tigger0.feed("grass"));

    MockTigger mock_tigger;
    FeedTigger feed_tigger1(&mock_tigger);
    ON_CALL(mock_tigger, eat(_))
    	.Times(1)
        .WillRepeatedly(::testing::Return(0));
    EXPECT_TRUE(feed_tigger1.feed("grass"));
}
}

可以看到,第一次在传入"grass"这个参数时,程序必定返回false,而第二次,我们使用了mock之后的结果,也就是说无论传入什么参数,返回的结果都会是true。

3. makefile文件

bash 复制代码
CXX = g++
CXXFLAGS = -Wall
LIBES = -lgtest -lgmock -lgtest_main -lpthread
LPATH = -L/tools/googletest/1.11.0/build/lib  # 替换成自己lib路径
HPATH = -I/tools/googletest/1.11.0/googletest/include/ # 替换成自己的
include路径
HPATH += -I/tools/googletest/1.11.0/googlemock/include/ # 替换成自己的
include路径

UTEST_OBJD = hello_unit_test

hello_unit_test:hello_unit_test.cpp
	${CXX} -o $@ $+ -I ../ ${HPATH} ${CXXFLAGS} ${LIBES} ${LPATH}

clean:
	rm -rf *_unit_test

gmock常用宏

1. MOCK_METHODx

其中x表示参数个数,如:

cpp 复制代码
MOCK_METHOD0(func1, void());
MOCK_METHOD1(func2, int(int));
MOCK_METHOD2(func3, int(int, int));

第一个参数,funcx表示函数名,第二个参数为函数返回值和参数列表。若是mock一个模板类的成员函数,只需要使用MOCK_METHODx_T即可。

2. ON_CALL

cpp 复制代码
ON_CALL(mock, func3(_, 1))
	.Times(4)
	.WillOnce(Return(0))
	.WillOnce(Return(1))
	.WillRepeatedly(Return(2));
  • func3中的参数:
    • '_'表示匹配任意参数
    • 1,表示匹配指定参数,即参数为1时,才能匹配,否则不会调用mock方法
  • Times,表示mock方法执行多少次
  • WillOnce,指定一次返回结果
  • WillRepeatedly,指定每次返回同样的结果
  • 上述指定mock结果之后,程序会在第一次调用func2方法时返回0,第二次返回1,第三次和第四次都返回2,第五次及之后返回真实结果。
  • 以上为基本用法,更多好玩的用法请参考官方文档

3. EXPECT_CALL

EXPECT_CALL宏在使用上与ON_CALL宏区别不大,可以参照其进行使用,主要区别在于EXPECT_CALL宏在指定调用次数(Times)后,如指定了调用3次,程序必须调用被mock的函数,使得mock结果生效3次,否则就会报测试失败或异常,而ON_CALL则不会。所以当确实需要mock结果生效指定次数时,那么就选择使用EXPECT_CALL。

总结

  • 本篇主要讲解了如何快速入门gmock以及基本语法。
  • gmock原理其实就是利用了多态的特性,通过继承需要被mock的类,然后重写其虚函数,在使用时,将被mock的对象替换成mock后的对象,这样在运行时就可以调用mock后的对象方法,从而达到测试预期。
  • 因此gmock本身只能mock虚函数,而对于non-virtual函数,如非虚成员函数、静态成员函数、全局函数、外部依赖库的函数等,该如何进行mock呢?我们将在下一篇章来介绍。
相关推荐
SUN_Gyq2 分钟前
什么是 C++ 中的模板特化和偏特化? 如何进行模板特化和偏特化?
开发语言·c++·算法
愿天垂怜12 分钟前
【C++】C++11引入的新特性(1)
java·c语言·数据结构·c++·算法·rust·哈希算法
大帅哥_18 分钟前
访问限定符
c语言·c++
小林熬夜学编程1 小时前
【Linux系统编程】第五十弹---构建高效单例模式线程池、详解线程安全与可重入性、解析死锁与避免策略,以及STL与智能指针的线程安全性探究
linux·运维·服务器·c语言·c++·安全·单例模式
凯子坚持 c1 小时前
C++之二叉搜索树:高效与美的极致平衡
开发语言·c++
埋头编程~1 小时前
【C++】踏上C++学习之旅(十):深入“类和对象“世界,掌握编程黄金法则(五)(最终篇,内含初始化列表、静态成员、友元以及内部类等等)
java·c++·学习
亚图跨际1 小时前
MATLAB和C++及Python流式细胞术
c++·python·matlab·流式细胞术
海绵波波1071 小时前
C++11:多线程编程
c++
程序猿阿伟2 小时前
《进程隔离机制:C++多进程编程安全的坚固堡垒》
服务器·c++·安全
gkdpjj2 小时前
C++优选算法十四 优先级队列(堆)
开发语言·数据结构·c++·算法