仿RabbiteMq简易消息队列基础篇(gtest的使用)

news2025/1/12 23:11:20
@TOC

gtest介绍

        gtest是google的一个开源框架,它主要用于写单元测试,检查自己的程序是否符合预期行为。可在多个平台上使用(包含Linux,MAC OC,Windows等)。它提供了丰富的断言,致命和非致命失败判断,能进行值参数化测试,类型参数化测试,“死亡测试”。

断言

        一般的要测试一个方法(函数)是否是正常执行的,可以提供一些输入数据,在调用这个函数后,会得到输出结果,然后检查输出的数据是否与我们期望的结果是一致的,若一致则说明这个方法的逻辑是正确的,否则就有问题。

        在对输出结果进行检查的时候,gtest为我们提供了一系列的断言进行代码测试,这些宏有点类似于函数调用。当断言失败时gtest将会打印出assertion时的源文件和出错行的位置以及附加的失败信息。这些输出的附加信息用户可以直接通过operator << 在这些断言宏后面。

        gtest中,断言的宏可以分为两种,一类是ASSERT系列,一类是EXPECT类。

        区别:ASSERT不通过的时候会认为是一个指明的错误,退出当前函数(只是函数)。而EXPECT失败之后会继续运行当前函数,所以对于函数内的几个失败可以同时报告出来,通常用我们EXPECT级别的断言即可,除非你认为当前检查点失败后函数的后续检查没有意义。

布尔类型检查

致命断言非致命断言验证
ASSERT_TRUE(状态)EXPECT_TRUE(状态)条件为真
ASSERT_FALSE(状态)EXPECT_FALSE(状态)条件为假

二值检查

致命断言非致命断言验证
ASSERT_EQ(期望值, 实际值)EXPERT_EQ(期望值, 实际值)期望值 == 实际值
ASSERT_NE(值1, 值2)EXPERT_NE(值1, 值2)值1 != 值2
ASSERT_LT(值1, 值2)EXPERT_LT(值1, 值2)值1 < 值2
ASSERT_LE(值1,值2)EXPERT_LE(值1, 值2)值1 <= 值2
ASSERT_GT(值1, 值2)EXPERT_GT(值1, 值2)值1 > 值2
ASSERT_GE(值1,值2)EXPERT_GE(值1, 值2)值1 >= 值2

字符串检查

致命断言非致命断言

验证

ASSERT_STREQ(字符串1, 字符串2)EXPECT_STREQ(字符串1, 字符串2)两个C字符串具有相同的内容
ASSERT_STRNE(字符串1, 字符串2)EXPECT_STRNE(字符串1, 字符串2)两个C字符串有不同的内容
ASSERT_STRCASEEQ(字符串1, 字符串2)EXPERT_STRCASEEQ(字符串1, 字符串2)两个C字符串具有相同的内容(忽略大小写)
ASSERT_STRCASENE(字符串1, 字符串2)EXPECT_STRCASENE(字符串1, 字符串2)两个C字符串的内容不同(忽略大小写)

宏测试

TEST宏

        TEST宏的第一个参数是test_case_name(测试套件名),第二个参数是test_name(测试特例名)

        测试套件(Test Case)是为某个特殊目标而编制的一组测试输入,执行条件以及预期结果,以便测试某个程序路径或核实是否满足某个特定需求。

        测试特例名是测试套件下的一个测试

举例:

#include <iostream>
#include <gtest/gtest.h>

using namespace std;

TEST(MATHTEST, test1)
{
    ASSERT_EQ(10, 10);     // 成功
    EXPECT_NE(10, 10);     // 失败,但是会继续执行
    ASSERT_TRUE(10 > 0);   // 成功
    EXPECT_FALSE(10 < 1);  // 成功
}

int main(int argc, char* argv[])
{
    testing::InitGoogleTest(&argc, argv);

    return RUN_ALL_TESTS();
}

TEST_F宏

        使用TEST_F前需要创建一个固定类,继承testing::Test类

        在类内部使用public或者protected描述其成员,为了保证实际执行的测试子类可以使用其成员变量。在构造函数或者继承于::testing::Test类中的SetUp方法中可以实现我们需要构造的数据。在析构函数或者继承于::testing::Test类中的TearDown方法中可以实现一些资源释放的代码。

        这里需要注意四个函数

//测试环境

测试夹具类中可以实现四个函数

static void SetUpTestCase()  //在测试某个测试套件时,在执行第一个测试用例前执行

static void TearDownTestCase() //在测试某个测试套件时,在执行完最后一个测试用例后执行

virtual void SetUp() override //在测试某个测试套件时,在执行每一个测试用例前执行

virtual void TearDown() override //在测试某个测试套件前,在执行完每一个测试用例后执行

        第一个参数为测试套件名(必须与创建的固件类名一致),第二个为测试名,可任意取。

        TEST_F宏和TEST宏的实现接近,只是TEST_F宏的封装更加开放一些,对TEST宏的功能多了一些扩展。

        TEST_F与TEST的区别,TEST_F提供了一个初始化函数(SetUp)和一个清理函数(TearDown)。在TEST_F中使用的变量可以在初始化函数SetUp中初始化,在TearDown中销毁。所有的TEST_F是互相独立的,都是在初始化以后的状态开始运行。一个TEST_F不会影响另一个TEST_F所使用的数据,多个测试场景需要相同数据配置的情况用TEST_F。

举例:

#include <iostream>
#include <gtest/gtest.h>
#include <vector>
#include <algorithm>

using namespace std;

vector<int> global_vector;

class VectorTest : public testing::Test
{
public:
    static void SetUpTestCase()
    {
        cout << " VectorTest测试套件开始测试" << endl;
        global_vector = {9, 5, 6, 2, 8, 7, 1, 3, 4};
    }
    static void TearDownTestCase()
    {
        cout << "VectorTest测试套件结束测试,清理数据" << endl;
        global_vector.clear();
    }
    void SetUp() override
    {
        cout << "VectorTest测试套件中新的一场测试开始测试" << endl;
        global_vector = {9, 5, 6, 2, 8, 7, 1, 3, 4};
    }

    void TearDown() override
    {
        cout << "VectorTest测试套件中旧的一场测试结束测试,清理数据" << endl;
        global_vector.clear();
    }
};

TEST_F(VectorTest, sortVector)
{
    sort(global_vector.begin(), global_vector.end());
    for(int i = 0; i < global_vector.size(); i++)
    {
        cout << global_vector[i] << " ";
    }
    cout << endl;
}

TEST_F(VectorTest, reductVector)
{
    for(int i = 0; i < global_vector.size(); i++)
    {
        cout << global_vector[i] << " ";
    }
    cout << endl;
}

TEST(MYTEST, test1)
{
    if(global_vector.empty())
    {
        cout << "数组已经清空" << endl;
    }
    else
    {
        cout << "数组没有清空" << endl;
    }
}

int main(int argc, char* argv[])
{
    testing::InitGoogleTest(&argc, argv);

    return RUN_ALL_TESTS();
}

TEST_P宏

        在设计测试案例时,经常需要考虑给被测函数传入不同的值的情况。我们之前的做法通常是写一个通用方法然后编写在测试案例调用它。即使使用了通用方法,这样的工作也是有很多重复性的。

        用TEST这个宏,需要编写如下的测试案例,每输入一个值就需要写一个测试点,这还只是在一个测试中,如果把每个测试点单独创建一个测试,工作量就更大。使用TEST_P这个宏,对输入进行参数化,就简单很多

举例:

#include <iostream>
#include <gtest/gtest.h>
#include <vector>
#include <algorithm>

using namespace std;

class positiveTest : public testing::TestWithParam<int>
{
public:
    static void SetUpTestCase()
    {
        cout << " VectorTest测试套件开始测试" << endl;
    }
    static void TearDownTestCase()
    {
        cout << "VectorTest测试套件结束测试,清理数据" << endl;
    }
    void SetUp() override
    {
        cout << "VectorTest测试套件中新的一场测试开始测试" << endl;
    }

    void TearDown() override
    {
        cout << "VectorTest测试套件中旧的一场测试结束测试,清理数据" << endl;
    }
};

class negativeTest : public testing::TestWithParam<int>
{
public:
    static void SetUpTestCase()
    {
        cout << " VectorTest测试套件开始测试" << endl;
    }
    static void TearDownTestCase()
    {
        cout << "VectorTest测试套件结束测试,清理数据" << endl;
    }
    void SetUp() override
    {
        cout << "VectorTest测试套件中新的一场测试开始测试" << endl;
    }

    void TearDown() override
    {
        cout << "VectorTest测试套件中旧的一场测试结束测试,清理数据" << endl;
    }
};

INSTANTIATE_TEST_SUITE_P
(
    positiveValues,
    positiveTest,
    testing::Values(9, 5, 6, 2, 8, 7, 1, 3, 4)
);

INSTANTIATE_TEST_SUITE_P
(
    negativeValues,
    negativeTest,
    testing::Values(-9, -5, -6, -2, -8, -7, -1, -3, -4)
);

TEST_P(positiveTest, Test1)
{
    int param = GetParam();
    EXPECT_GT(param, 0);

}



TEST_P(negativeTest, Test1)
{
    int param  = GetParam();
    EXPECT_LT(param, 0);

}


int main(int argc, char* argv[])
{
    testing::InitGoogleTest(&argc, argv);

    return RUN_ALL_TESTS();
}

预处理事件机制

gtest 提供了多种预处理事件机制,方便我们在测试之前或之后做一些操作。

1. 全局的,所有测试执行前后。

2. TestSuite级别的,在某测试套件中第一个测试前,最后一个测试执行后。

3. TestCase级别的,每个测试前后。

1.全局事件

        要实现全局事件,必须写一个类继承testing::Environment类,实现里面的SetUp和TearDown方法。

        1. SetUp()方法在所有案例执行前执行。

        2. TearDown()方法在所有案例执行后执行。

        还需要在main函数中通过调用testing::AddGlobalTestEnvironment这个函数将事件挂进来,也就是说,我们可以写很多个这样的类,然后将他们的事件都挂上去,AddGlobalTestEnvironment这个函数要放在RUN_ALL_TEST之前。

int main()
{
    testing::AddGlobalTestEnvironment(new FooEnvir);
    testing::InitGoogleTest();
    RUN_ALL_TESTS();
    return 0;
}

2.TestSuites事件

        需要写一个类,继承testing::Test,然后实现两个静态方法

        1. SetUpTestCase() 方法在第一个TestCase之前执行。

        2. TearDownTestCase() 方法在最后一个TestCase之后执行。

3.TestCase事件

        TestCase事件是挂在每个案例执行前后的,实现方式和Test'Suites的几乎一样,不过需要实现的是SetUp方法和TearDown方法:

1. SetUp()方法在每个TestCase之前执行。

2. TearDown()方法在每个TestCase之后执行。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1996781.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Spring框架的三种配置方式(二)---xml文件+注解

Spring框架有三种配置方式&#xff1a; 1.在spring2.5以前&#xff0c;用xml文件进行配置 2.在spring2.5以后&#xff0c;用xml文件和注解(annotation)共同进行配置 3.在spring3.0以后&#xff0c;用注解(annotation)和JavaConfig配置类进行配置 一、xml文件 见下一篇 二…

联通数科如何基于Apache DolphinScheduler构建DataOps一体化能力平台

各位小伙伴晚上好&#xff0c;我是联通数字科技有限公司数据智能事业部的王兴杰。 今天&#xff0c;我将和大家聊一聊联通数字科技有限公司是如何基于Apache DolphinScheduler构建DataOps一体化能力平台的。 今天的分享主要分为三个部分&#xff1a; 关于DataOps的一些思考&a…

视觉SLAM第五讲

本讲将讨论“机器人如何观测外部世界”&#xff0c;也就是观测方程部分。而在以相机为主的视觉SLAM中&#xff0c;观测主要是指相机成像的过程。 三维世界中的一个物体反射或发出的光线&#xff0c;穿过相机光心后&#xff0c;投影在相机的成像平面上。相机的感光器件接收到光…

主机加固是什么?主机加固与产线工控安全关系

1. 需求背景 随着工业4.0的发展&#xff0c;生产线日益智能化&#xff0c;生产网已经发展成一个复杂的计算机环境。尽管这些网络通常进行了物理隔离&#xff0c;但在实际操作中仍需要与外部进行数据交互。这种交互可能导致病毒和恶意软件的入侵&#xff0c;威胁工控主机和产线…

k8s分布式存储-ceph

文章目录 Cephdeploy-ceph部署1.系统环境初始化1.1 修改主机名&#xff0c;DNS解析1.2 时间同步1.3 配置apt基础源与ceph源1.4关闭selinux与防火墙1.5 **创建** ceph **集群部署用户** cephadmin1.6分发密钥 2. ceph部署2.1 **安装** ceph 部署工具2.2 **初始化** mon **节点**…

计算机毕业设计选题推荐-小型民营加油站管理系统-Java/Python项目实战

✨作者主页&#xff1a;IT研究室✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Python…

使用 Java Swing 创建一个最大公约数计算器 GUI 应用

使用Java语言,设计一个程序,实现求取两个正整数的最大公约数。 比较基础的一个Java小程序。 1、效果展示 2、程序代码 package demo; import javax.swing.*; import java.awt.*;

用python连接mysql的方法

如何将个人主机上的mysql服务发布到公网&#xff1a;frp内网穿透 用python连接mysql的方法 方法一&#xff1a; 1.设置清华镜像站&#xff1a;从国内下载安装包&#xff0c;提高下载和安装速度 pip3 config set global.index-url https://pypi.tuna.tsinghua.edu.cn/simp…

航向角、前轮转角、偏航角的区别及其在MPC中的使用

目录 前言一、概念解析二、三种角度在MPC中的应用三、总结四、MPC算法流程 前言 航向角、偏航角、前轮转角是车辆控制中描述方向的关键概念。本文将简要介绍它们的区别及在MPC&#xff08;模型预测控制&#xff09;中的应用。 一、概念解析 φ 为车体的航向角&#xff1b; δ…

[ACP云计算]组件介绍

一、IaaS、PaaS、SaaS 二、交换机 三、VPC 四、ECS 云服务器ECS&#xff08;Elastic Compute Service&#xff09;是阿里云提供的性能卓越、稳定可靠、弹性扩展的IaaS&#xff08;Infrastructure as a Service&#xff09;级别云计算服务。云服务器ECS免去了您采购IT硬件的前期…

jupyter项目使用Anaconda环境内核

1、创建虚拟环境 conda create --name myjupyter python3.7 2、进入虚拟环境 conda activate myjupyter 3、切换到自己jupyter notebook项目想在的目录 E: cd E:\first\project\jupyter\jupyter01 4、安装IPython内核包&#xff0c;这是Jupyter Notebook使用Python内核所必需的…

【MySQL 03】库的操作 (带思维导图)

前置&#xff1a;之后的所有的 SQL 语句中&#xff0c;凡是被中括号 [ ] 括起来的均为可选项。 &#x1f308; 一、创建数据库 数据库创建语句 创建数据库本质就是在 /var/lib/mysql 中创建一个目录。 if not exists&#xff1a;如果指定数据库不存在则创建该数据库&#xf…

[Java]面向对象-static继承

Static static表示静态&#xff0c;是Java中的一个修饰符&#xff0c;可以修饰成员方法、成员变量 静态变量&#xff1a; 被static修饰的成员变量&#xff0c;叫静态变量 特点&#xff1a; 被该类所有对象共享 调用方式&#xff1a;1.类名调用 2.对象名调用 静态方法&…

PyFluent入门之旅(8) PyFluent API 分类与区别

PyFluent提供了两种主要的API来与Ansys Fluent进行交互&#xff1a; Settings APITUI API 通过这两种接口方式&#xff0c;可以控制 Ansys Fluent 的各个方面&#xff0c;包括从网格生成到后处理的所有操作。 分类 Settings API pyFluent 的 Settings API类似于 Ansys Flu…

线性表——数据结构

线性表 文章目录 线性表线性表的定义和基本操作线性表的定义线性表的基本操作 线性表的顺序表示顺序表的定义顺序表的实现——静态分配顺序表的实现——动态分配顺序表的特点 线性表的定义和基本操作 线性表的定义 线性表&#xff08;Linear List&#xff09;的定义 ​ 线性…

略谈set与map的pair封装与进入哈希

引子&#xff1a;之前我们讲了红黑树的自实现&#xff0c;与小小的接口实现&#xff0c;那set与map的pair封装是如何实现的呢&#xff1f;&#xff0c;今天我们来一探究竟&#xff0c;而且我们也要进入新章节--哈希 对于operator--()的封装&#xff1a; 注意&#xff1a;牢记思…

动手学深度学习V2每日笔记(批量归一化、ResNet)

本文主要参考沐神的视频教程 https://www.bilibili.com/video/BV1Uv411G71b/spm_id_fromautoNext&vd_sourcec7bfc6ce0ea0cbe43aa288ba2713e56d https://cv.gluon.ai/model_zoo/classification.html 文档教程 https://zh-v2.d2l.ai/ 本文的主要内容对沐神提供的代码中个人不…

【OpenHarmony】openharmony移植到RK3568------搭建开发环境

一、关于OpenHarmony OpenHarmony是由开放原子开源基金会&#xff08;OpenAtom Foundation&#xff09;孵化及运营的开源项目&#xff0c;目标是面向全场景、全连接、全智能时代&#xff0c;基于开源的方式&#xff0c;搭建一个智能终端设备操作系统的框架和平台&#xff0c;促…

各种高端链游 区块链游戏 休闲小游戏DAPP开发

【西游闯关】-高端区块链3D手游 【我朝有马】-高端区块链3D手游 【弹弹岛2】-高端区块链3D手游

FiddlerCharles请求包导入Postman教程

前言 晓杰因开发需要经常需要使用抓包工具进行抓包然后使用POSTMAN测试&#xff0c;每次复制都很麻烦&#xff0c;就想着如何直接抓包工具中导出&#xff0c;postman导入&#xff01; Fiddler 这个有2个方案&#xff0c;第一个方案是因为第二个方案发现的&#xff01; 方案…