当前位置: 首页 > news >正文

gtest框架的安装与使用

1、介绍

GTest 是一个跨平台的 C++单元测试框架,由 google 公司发布。gtest 是为了在不同 平台上为编写 C++单元测试而生成的。它提供了丰富的断言、致命和非致命判断、参数化等等测试所需的宏,以及全局测试,单元测试组件。

2、安装

sudo apt-get install libgtest-dev

3、使用

头文件的包含

#include <gtest/gtest.h>

框架初始化接口

testing::InitGoogleTest(&argc, argv);

调用测试样例

RUN_ALL_TESTS();

TEST宏

//这里不需要双引号,且同测试下多个测试样例不能同名

TEST(测试名称, 测试样例名称)

TEST_F(test_fixture,test_name)

  • TEST:主要用来创建一个简单测试, 它定义了一个测试函数, 在这个函数中可以使用任何 C++代码并且使用框架提供的断言进行检查
  • TEST_F:主要用来进行多样测试,适用于多个测试场景如果需要相同的数据配置的情况, 即相同的数据测不同的行为

断言宏

GTest 中的断言的宏可以分为两类:
  • ASSERT_系列:如果当前点检测失败则退出当前函数
  • EXPECT_系列:如果当前点检测失败则继续往下执行
下面是经常使用的断言介绍

// bool 值检查

ASSERT_TRUE(参数),期待结果是 true

ASSERT_FALSE(参数),期待结果是 false

//数值型数据检查

ASSERT_EQ(参数 1,参数 2),传入的是需要比较的两个数 equal

ASSERT_NE(参数 1,参数 2),not equal,不等于才返回 true

ASSERT_LT(参数 1,参数 2),less than,小于才返回 true

ASSERT_GT(参数 1,参数 2),greater than,大于才返回 true

ASSERT_LE(参数 1,参数 2),less equal,小于等于才返回 true

ASSERT_GE(参数 1,参数 2),greater equal,大于等于才返回 true

样例

main.cc

#include <gtest/gtest.h>int abs(int x)
{return x > 0 ? x : -x;
}TEST(abs_test, test1)
{ASSERT_TRUE(abs(1) == 1) << "abs(1)=1";ASSERT_TRUE(abs(-1) == 1);ASSERT_FALSE(abs(-2) == -2);ASSERT_EQ(abs(1),abs(-1));ASSERT_NE(abs(-1),0);ASSERT_LT(abs(-1),2);ASSERT_GT(abs(-1),0);ASSERT_LE(abs(-1),2);ASSERT_GE(abs(-1),0);
}int main(int argc, char* argv[])
{//单元测试框架的初始化testing::InitGoogleTest(&argc, argv);//开始所有的单元测试return RUN_ALL_TESTS();
}

makefile

main: main.ccg++ -std=c++17 $^ -o $@ -lgtest

运行结果

事件机制

GTest 中的事件机制是指在测试前和测试后提供给用户自行添加操作的机制,而且该机制也可以让同一测试套件下的测试用例共享数据。GTest 框架中事件的结构层次:

测试程序:一个测试程序只有一个 main 函数,也可以说是一个可执行程序是一个测试程序。该级别的事件机制是在程序的开始和结束执行

  • 测试套件:代表一个测试用例的集合体,该级别的事件机制是在整体的测试案例开始和结束执行
  • 测试用例:该级别的事件机制是在每个测试用例开始和结束都执行

事件机制的最大好处就是能够为我们各个测试用例提前准备好测试环境,并在测试完毕后用于销毁环境,这样有个好处就是如果我们有一端代码需要进行多种不同方法的测试,则可以通过测试机制在每个测试用例进行之前初始化测试环境和数据,并在测试完毕后清理测试造成的影响。

GTest 提供了三种常见的的事件:

全局事件

针对整个测试程序。实现全局的事件机制,需要创建一个自己的类,然后继承testing::Environment 类,然后分别实现成员函数 SetUp TearDown,同时在 main 函数内进行调用testing::AddGlobalTestEnvironment(new MyEnvironment);函数添加全局的事件机制

#include <iostream> 
#include <gtest/gtest.h>
//全局事件:针对整个测试程序,提供全局事件机制,能够在测试之前配置测试环境
数据,测试完毕后清理数据
//先定义环境类,通过继承 testing::Environment 的派生类来完成
//重写的虚函数接口 SetUp 会在测试之前被调用; TearDown 会在测试完毕后调
用. 
std::unordered_map<std::string, std::string> dict;
class HashTestEnv : public testing::Environment {public:virtual void SetUp() override{std::cout << "测试前:提前准备数据!!\n";dict.insert(std::make_pair("Hello", "你好"));dict.insert(std::make_pair("hello", "你好"));dict.insert(std::make_pair("雷吼", "你好"));} virtual void TearDown() override{std::cout << "测试结束后:清理数据!!\n";dict.clear();} 
};
TEST(hash_case_test, find_test) {auto it = dict.find("hello");ASSERT_NE(it, dict.end());
} 
TEST(hash_case_test, size_test) {ASSERT_GT(dict.size(), 0);
} 
int main(int argc, char *argv[])
{ testing::AddGlobalTestEnvironment(new HashTestEnv );testing::InitGoogleTest(&argc, argv);return RUN_ALL_TESTS();
}

TestSuite事件

针对一个个测试套件。测试套件的事件机制我们同样需要去创建一个类,继承自 testing::Test,实现两个静态函数 SetUpTestCase TearDownTestCase,测试套件的事件机制不需要像全局事件机制一样在 main 注册,而是需要将我们平时使用的 TEST 宏改为 TEST_F 宏。

  • SetUpTestCase() 函数是在测试套件第一个测试用例开始前执行
  • TearDownTestCase() 函数是在测试套件最后一个测试用例结束后执行
  • 需要注意 TEST_F 的第一个参数是我们创建的类名,也就是当前测试套件的名称,这样在 TEST_F 宏的测试套件中就可以访问类中的成员了。
#include <iostream>
#include <gtest/gtest.h>
//TestSuite:测试套件/集合进行单元测试,即,将多个相关测试归入一组的方式进
行测试,为这组测试用例进行环境配置和清理
//概念: 对一个功能的验证往往需要很多测试用例,测试套件就是针对一组相关测
试用例进行环境配置的事件机制
//用法: 先定义环境类,继承于 testing::Test 基类, 重写两个静态函数
SetUpTestCase/TearDownTestCase 进行环境的配置和清理
class HashTestEnv1 : public testing::Test {public:static void SetUpTestCase() {std::cout << "环境 1 第一个 TEST 之前调用\n";} static void TearDownTestCase() {std::cout << "环境 1 最后一个 TEST 之后调用\n";} public:std::unordered_map<std::string, std::string> dict;
};
// 注意,测试套件使用的不是 TEST 了,而是 TEST_F, 而第一个参数名称就是测试
套件环境类名称
// main 函数中不需要再注册环境了,而是在 TEST_F 中可以直接访问类的成员变量
和成员函数
TEST_F(HashTestEnv1, insert_test) {std::cout << "环境 1,中间 insert 测试\n";dict.insert(std::make_pair("Hello", "你好"));dict.insert(std::make_pair("hello", "你好"));dict.insert(std::make_pair("雷吼", "你好"));auto it = dict.find("hello");ASSERT_NE(it, dict.end());
} 
TEST_F(HashTestEnv1, sizeof) {std::cout << "环境 1,中间 size 测试\n";ASSERT_GT(dict.size(), 0);
} int main(int argc, char *argv[])
{ testing::InitGoogleTest(&argc, argv);return RUN_ALL_TESTS();
}

结构

能够看到在上例中,有一个好处,就是将数据与测试结合到同一个测试环境类中了,这样与外界的耦合度更低,代码也更清晰。

但是同样的,我们发现在两个测试用例中第二个测试用例失败了,这是为什么呢?这就涉及到了 TestCase 事件的机制。

TestCase 事件

针对一个个测试用例。测试用例的事件机制的创建和测试套件的基本一样,不同地方在于测试用例实现的两个函数分别是 SetUp TearDown, 这两个函数也不是静态函数

  • SetUp()函数是在一个测试用例的开始前执行
  • TearDown()函数是在一个测试用例的结束后执行

也就是说,在 TestSuite/TestCase 事件中,每个测试用例,虽然它们同用同一个事件环境类,可以访问其中的资源,但是本质上每个测试用例的环境都是独立的,这样我们就不用担心不同的测试用例之间会有数据上的影响了,保证所有的测试用例都使用相同的测试环境进行测试。

#include <iostream>
#include <gtest/gtest.h>
//TestCase:测试用例的单元测试,即针对每一个测试用例都使用独立的测试环境数据进行测试
//概念:它是针对测试用例进行环境配置的一种事件机制
//用法:先定义环境类,继承于 testing::Test 基类, 在环境类内重写SetUp/TearDown 接口
class HashTestEnv2 : public testing::Test {public:static void SetUpTestCase() {std::cout << "环境 2 第一个 TEST 之前被调用,进行总体环境配置\n";} static void TearDownTestCase() {std::cout << "环境 2 最后一个 TEST 之后被调用,进行总体环境清理\n";} virtual void SetUp() override{std::cout << "环境 2 测试前:提前准备数据!!\n";dict.insert(std::make_pair("bye", "再见"));dict.insert(std::make_pair("see you", "再见"));} virtual void TearDown() override{std::cout << "环境 2 测试结束后:清理数据!!\n";dict.clear();}public:std::unordered_map<std::string, std::string> dict;
};TEST_F(HashTestEnv2, insert_test) {std::cout << "环境 2,中间测试\n";dict.insert(std::make_pair("hello", "你好"));ASSERT_EQ(dict.size(), 3);
} 
TEST_F(HashTestEnv2, size_test) {std::cout << "环境 2,中间 size 测试\n";auto it = dict.find("hello");ASSERT_EQ(it, dict.end());ASSERT_EQ(dict.size(), 2);
} 
int main(int argc, char *argv[])
{ testing::InitGoogleTest(&argc, argv);RUN_ALL_TESTS();return 0;
}

运行结果:

http://www.lryc.cn/news/604887.html

相关文章:

  • C#程序员计算器
  • 单片机学习笔记.AD/DA(略含有SPI,用的是普中开发板上的XPT2046芯片)
  • Rust × Elasticsearch官方 `elasticsearch` crate 上手指南
  • 《安富莱嵌入式周报》第356期:H7-TOOL的250M示波器模组批量生产中,自主开发QDD执行器,开源14bit任意波形发生器(2025-07-28)
  • ConcurrentHashMapRedis实现二级缓存
  • (LeetCode 面试经典 150 题) 141. 环形链表(快慢指针)
  • 如何将JPG、PNG、GIF图像转换成PDF、SVG、EPS矢量图像
  • 简单线性回归模型原理推导(最小二乘法)和案例解析
  • react+ant design怎么样式穿透-tooltip怎么去掉箭头
  • 工作笔记-----存储器类型相关知识
  • Solon v3.4.2(Java 应用开发生态基座)
  • Java 控制台用户登录系统(支持角色权限与自定义异常处理)
  • python之asyncio协程和异步编程
  • 【MySQL学习|黑马笔记|Day3】多表查询(多表关系、内连接、外连接、自连接、联合查询、子查询),事务(简介、操作、四大体系、并发事务问题、事务隔离级别)
  • 自动化与配置管理工具 ——Ansible
  • 创建型设计模式-Builder
  • Newman+Jenkins实施接口自动化测试
  • 浏览器pdf、image显示
  • 防火墙与入侵检测
  • Linux下C/C++服务端崩溃排查手册
  • VMware16安装Ubuntu-22.04.X版本(并使用桥接模式实现局域网下使用ssh远程操作Ubuntu系统)
  • Linux 进程管理与计划任务设置
  • 16-C语言:第17天笔记
  • UnityHub Validation Failed下载编辑器错误,添加模块报错的解决方案
  • 外网访问文档编辑器Docsify(Windows版本),内网穿透技术应用简便方法
  • 论文阅读|ArxiV 2024|Mamba进一步研究|VSSD
  • Linux虚拟内存
  • Vision Transformer (ViT) 详解:当Transformer“看见”世界,计算机视觉的范式革命
  • LeetCode 160:相交链表
  • 【推荐100个unity插件】Unity 创意编程库——Klak插件的使用