首页 > 编程语言 >C/C++单元测试如何解决非虚函数对象依赖

C/C++单元测试如何解决非虚函数对象依赖

时间:2024-06-11 13:32:18浏览次数:30  
标签:非虚 函数 CppFreeMock 单元测试 C++ MOCKER expectValues receivedatafromdevice

如何解决非虚函数对象依赖

随着事物的接触越来越多,了解的越来越深入,我们总会发现一些新的问题或者不足。

就像前文提到的一样,我们在面对有对象的虚函数依赖的时候,可以使用gmock框架来为我们提供方便的模拟期望值,以便我们能撇除外界的影响(依赖)从逻辑上设计单元测试并持续的进行,但是并非所有对象的函数都设计成了虚函数,那么我们在面对依赖对象的非虚函数这个问题时,又该如何解决?

这个问题,已经有先行者遇到并且提出了解决方案:手动打桩、使用hook技术。

手动打桩有一个stub挺好用,只有一个头文件,包含进去就可以使用,但由于手动,所以使用起来相对有一些繁琐,并且不能很好的统计和校验调用次数。

使用hook技术的有mockcppCppFreeMock,这里使用的是CppFreeMock。因为它是基于gmock而来,是对gmock只能mock虚函数的一个补充,并且在用法上也能完美兼容gmock框架,如果单元测试已经是使用gtest+gmock的组合了,那么使用CppFreeMock的成本将不会高。

举个栗子:

void model::hardwardResponse() {
    memset(m_aRecvResponse, 0, MAXLEN);
    int iRet = _pdevice->receivedatafromdevice(m_aRecvResponse);
    if (iRet == 0)
    {
        m_iHardwareErrCode = ERR_RECV_INVALID_LEN;
        return;
    }

    m_iHardwareErrCode = m_aRecvResponse[ERRNO];
    if (m_iHardwareErrCode == 0)
    {
        /* NO ERROR, response data process... */
    }
}

这个函数,依赖于设备返回的数据。并且receivedatafromdevice函数是一个非虚函数:

class device
{
private:
    string _serialno;
    string _version;
    string _firewareversion;
    devicetype _type;
    hardware* _phardware;

public:
    device();
    ~device();
    bool senddatatodevice(unsigned char* buff, int len);
    int receivedatafromdevice(unsigned char* receivedata);
    char* requestdeviceinfo(int requesttype);
    devicetype requestdevicetype();
};

我们不用修改原本已有的modelTest测试套件,需要先将CppFreeMock的头文件包含进来

#include "cpp_free_mock.h"

然后,开始设计我们的测试用例:

TEST_F(modelTest, hardwareResponse_Lenis0) {
	/* 测试硬件返回长度为0的情况 */
	// 准备动作
	// 执行函数   
	pm->hardwardResponse();
	// 校验期望
	EXPECT_EQ(pm->getHardwardCode(), 44444);
}

TEST_F(modelTest, hardwareResponse_Success) {
	/* 测试硬件数据正常的情况 */
	// 准备动作
	// 执行函数   
    pm->hardwardResponse();
	// 校验期望
    EXPECT_EQ(pm->getHardwardCode(), 0);
    EXPECT_EQ(memcmp(pm->getResponseData(), expectValues, 254), 0);
}

TEST_F(modelTest, hardwareResponse_Error) {
	/* 测试硬件数据异常的情况 */
	// 准备动作
	// 执行函数   
    pm->hardwardResponse();
	// 校验期望
    EXPECT_EQ(pm->getHardwardCode(), 4);
}

根据上面hardwardResponse函数的实现,设计出上述三个单元测试用例,那么准备工作该如何使用CppFreeMock来设定预期呢?下面以第二个测试用例作详细说明:


// 准备动作
unsigned char expectValues[1024];
for (int i = 0; i < 255; i++) {
    expectValues[i] = (i==4)?0:i;
}   

auto mockerDevice = MOCKER(&device::receivedatafromdevice);
EXPECT_CALL(*mockerDevice, MOCK_FUNCTION(_,_)).Times(1).WillOnce(DoAll(SetArrayArgument<1>(expectValues, expectValues+254), Return(32)));

auto mockerDevice = MOCKER(&device::receivedatafromdevice); // 创建device类的非虚成员函数receivedatafromdevicemock对象

auto : 这里用的autoC++11中的关键字

MOCKER : MOCKER宏是CppFreeMock中定义的,其作用是用于创建指定类的指定函数的mock对象。

普通成员函数用法:MOCKER(&类名::函数名) --例–>MOCKER(&device::receivedatafromdevice)
静态成员函数用法:MOCKER(类名::函数名) --例–>MOCKER(device::receivedatafromdevice)
普通全局函数用法:MOCKER(函数名) --例–>MOCKER(receivedatafromdevice)

更多用法请查阅 CppFreeMock

EXPECT_CALL(*mockerDevice, MOCK_FUNCTION(_, _)).Times(1)
.WillOnce(DoAll(SetArrayArgument<1>(expectValues, expectValues+254), Return(32))); // 期望receivedatafromdevice函数调用1次,传出的数据是expectValues数组中的前255个内容,并且返回接收数据长度为32个字节

MOCK_FUNCTION : 宏是CppFreeMock中定义的,表明是一个mock函数对象。
DoAll : 表明这次函数调用时,gmock需要执行DoAll(Action,Action)中的多个Action
SetArrayArgument(fisrtAddr, lastAddr) : 表示需要将从【firstAddr,lastAddr】 段中的数据传给第 n 个参数

其中TimesWillOnceReturn等的用法,[[5-如何解决虚函数对象依赖?|前文(函数有其他对象虚函数依赖如何单元测试)]] 有作说明,这里不多做赘述。

最前面的expectValues的数据定义及赋值,并且对错误码索引(4)特殊处理,赋值为0 [无错误]


完善好测试用例之后,运行看看CppFreeMock是否能解决我们非虚函数依赖的问题:
image.png

测试用例成功通过,表明非虚函数receivedatafromdevice按照我们设定的预期执行。

对应的demo源码,请点击 mocknonvirtualfunc

也可扫码关注博主同名公众号"不解之榬",回复 “非虚” 获取
不解之榬

标签:非虚,函数,CppFreeMock,单元测试,C++,MOCKER,expectValues,receivedatafromdevice
From: https://blog.csdn.net/LT450196683/article/details/139584082

相关文章

  • nw.js 如何调用activeX控件 (控件是C++编写的dll文件)
    ......
  • 深入理解C++中的常量和宏:const、#define、typedef和inline详解
    一、const 与 #define 的区别1.定义方式和类型const 定义的常量是有类型的变量。#define 只是文本替换,不带类型。constintMAX_VALUE=100;//MAX_VALUE是一个整数类型的常量#defineMAX_VALUE100//MAX_VALUE是一个文本替换,它不关联任何类型2.生效......
  • acwing 653 钞票(c++)
    题目描述:在这个问题中,你需要读取一个整数值并将其分解为多张钞票的和,每种面值的钞票可以使用多张,并要求所用的钞票数量尽可能少。请你输出读取值和钞票清单。钞票的可能面值有100,50,20,10,5,2,1。经过实验证明:在本题中,优先使用面额大的钞票可以保证所用的钞票总数量最......
  • c/c++ 设计模式-----职责链(Chain Of Responsibility)模式
    一个关于涨薪审批的范例#include<iostream>#ifdef_DEBUG//只在Debug(调试)模式下#ifndefDEBUG_NEW#defineDEBUG_NEWnew(_NORMAL_BLOCK,__FILE__,__LINE__)//重新定义new运算符#definenewDEBUG_NEW#endif#endif//#include<boost/type_index.hpp>usingnames......
  • c/c++设计模式---策略模式
    一个具体范例的逐步重构Fighter.h#ifndef__RIGHTER__#define__RIGHTER__////增加补充生命值道具(药品)//enumItemAddlife//{//LF_BXD,//补血丹//LF_DHD,//大还丹//LF_SHD,//守护丹//};classItemStrategy;//类前向声明//战斗者父类class......
  • c/c++设计模式---状态模式
    一个基本的状态转换范例monster.h#ifndef_MONSTER__#define_MONSTER__classMonsterStatus;//类前向声明//怪物类classMonster{public:Monster(intlife);~Monster();public:intGetLife()//获取怪物血量{returnm_life;}......
  • C++11 运算符的优先级分组和结合性
    本文汇总了C++11中的运算符的含义、优先级分组及其结合性。如果两个运算符用于同一个操作数,首先应用优先极高的。如果两个运算符优先级相同,按结合性规则决定应用那个运算符。同一组中的运算符优先级和结合性相同,无论先左后右(L-R),还是先右后左(R-L)。运算符含义优先级分组......
  • C++进阶教程
    一、引言C++是一种高效、强大且灵活的编程语言,广泛应用于系统软件开发、游戏开发、科学计算等领域。对于已经掌握C++基础知识的开发者来说,进阶学习C++将帮助他们更深入地理解这门语言,并提升编程能力。本教程将介绍C++中的一些高级特性和技术,包括面向对象编程、模板编程、ST......
  • OpenCV实战案例——校正+切边[C++]
    0.前言本文以实战案例为背景,讲述如何使用计算机图形学知识完成需求,实现最终效果。本文包含实战案例素材以及过程代码讲解,方便读者理解。1.案例需求某公司打算开发一款用于提取学生作业本的程序,学生用手机拍摄自己的作业上传到程序,程序进行处理最终提取出作业本区域方便老师批改......
  • 华为云短信服务教你用C++实现Smgp协议
    本文分享自华为云社区《华为云短信服务教你用C++实现Smgp协议》,作者:张俭。引言&协议概述中国联合网络通信有限公司短消息网关系统接口协议(SGIP)是中国网通为实现短信业务而制定的一种通信协议,全称叫做ShortMessageGatewayInterfaceProtocol,用于在短消息网关(SMG)和服务提供商(SP......