C++单元测试框架gtest使用
作用
- 作为代码编码人员,写完代码,不仅要保证编译通过和运行,还要保证逻辑尽量正确。单元测试是对软件可测试最小单元的检查和校验。单元测试与其他测试不同,单元测试可看作是编码工作的一部分,应该由程序员完成,也就是说,经过了单元测试的代码才是已完成的代码,提交产品代码时也要同时提交测试代码。测试部门可以作一定程度的审核。(来自百度百科)
其作用可以归为四种:
- 验证:验证程序逻辑的正确性,即使后期升级,通过跑单元测试,也可以看到升级后是否会对旧逻辑有影响
- 设计:促使程序员写出可单独测试的代码,从而更容易解耦
- 文档:作为程序使用的sample
- 回归:应对程序升级,也可以集成到code review之前的编译,自动做回归,典型用法就是在DevOps的编译打包阶段,例如jeckins编译完成后,自动触发单元测试。
- gtest是google提供的一套针对C++的单元测试框架,本文主要阐述其原理和使用方法。
- 与gtest配套的框架叫gmock,用于打桩,后续有空再写一篇介绍gmock的使用
原理
- gtest主要由一系列的宏和事件实现。
- 宏:有TEST和TEST_F宏,TEST宏针对简单的测试用例,TEST_F宏针对需要做初始化和资源回收的测试用例,有点像类似C++的构造函数和析构函数,两个宏都是把参数展开后拼成一个类。
- 事件:分为三种事件
-

测试程序:一个进程,全局事件在该层
测试套件:一系列测试用例的集合,SetUpTestCase事件在该层(注意,网上很多地方大部分写的是SetUpTestSuite,应该是老版本)
测试用例:一个测试用例的事件体现在每个用例前后的SetUp和TearDown
- 每个事件可用于下级时间的数据共享以及测试前后的数据处理
-
- gtest主要由一系列的宏和事件实现。
使用
- 如果时间充足,想整体了解,可以参考wiki
- 宏
- TEST
TEST(TestSuiteName, TestName) {
... statements ...
}会生成TestSuiteName_TestName_,不同的TestSuiteName可以有相同的TestName,需要注意的是所有的名字都不能有下划线
- TEST_F
TEST_F(TestFixtureName, TestName) {
... statements ...
}固定的套件名字
- TEST_P
TEST_P(MyTestSuite, DoesSomething) {
...
EXPECT_TRUE(DoSomething(GetParam()));
...
}支持传递参数
- TEST
- 事件
- 全局事件:需要创建一个自己的类,然后继承testing::Environment类,分别实现成员函数SetUp()和TearDown(),同时在main函数内进行调用,即"testing::AddGlobalTestEnvironment(new MyEnvironment);",通过调用函数我们可以添加多个全局的事件机制。需要注意的是,全局事件在--gtest_repeat参数大于1的情况下(可以通过--help查看其他参数),会执行多次
- 代码:

#include <iostream> #include <gtest/gtest.h> using namespace std; class MyEnvironment0 : public testing::Environment {
public:
virtual void SetUp() {
cout << "Global event0 : start1" << endl;
} virtual void TearDown() {
cout << "Global event0 : end" << endl;
}
}; class MyEnvironment1 : public testing::Environment {
public:
virtual void SetUp() {
cout << "Global event1 : start" << endl;
} virtual void TearDown() {
cout << "Global event1 : end" << endl;
}
}; TEST(GlobalTest0, test0) {
EXPECT_EQ(1, 1);
}; TEST(GlobalTest0, test1) {
EXPECT_EQ(1, 1);
}; TEST(GlobalTest1, test0) {
EXPECT_EQ(1, 1);
}; int main(int argc, char *argv[]) {
testing::AddGlobalTestEnvironment(new MyEnvironment0);
testing::AddGlobalTestEnvironment(new MyEnvironment1); testing::InitGoogleTest(&argc, argv); return RUN_ALL_TESTS();
}
- 代码:
- 套件事件:继承testing::Test,实现两个静态函数SetUpTestCase()和TearDownTestCase(),测试套件的事件机制不需要像全局事件机制一样在main注册,而是需要将我们平时使用的TEST宏改为TEST_F宏
- 代码:

#include <iostream> #include <gtest/gtest.h> using namespace std; class MyTestSuite0 : public testing::Test
{
protected:
// 网上大部分写的是SetUpTestSuite,google后面升级版本了,改成SetUpTestCase
// 相关讨论:https://stackoverflow.com/questions/54468799/google-test-using-setuptestsuite-doesnt-seem-to-work
static void SetUpTestCase()
{
cout << "TestSuite event0 : start" << endl;
} static void TearDownTestCase()
{
cout << "TestSuite event0 : end" << endl;
}
}; class MyTestSuite1 : public testing::Test
{
protected:
static void SetUpTestCase()
{
cout << "TestSuite event1 : start" << endl;
} static void TearDownTestCase()
{
cout << "TestSuite event1 : end" << endl;
}
}; // 必须用TEST实现,实现上是拼成一个类MyTestSuite0_test0
TEST_F(MyTestSuite0, test0)
{
EXPECT_EQ(1, 1);
} TEST_F(MyTestSuite1, test0)
{
EXPECT_EQ(1, 1);
} TEST_F(MyTestSuite0, test1)
{
EXPECT_EQ(1, 1);
} TEST_F(MyTestSuite1, test1)
{
EXPECT_EQ(1, 1);
} //int main(int argc, char *argv[])
//{
// testing::InitGoogleTEST_F(&argc, argv);
//
// return RUN_ALL_TESTS();
//}
- 代码:
- 用例事件:测试用例的事件机制的创建和测试套件的基本一样,不同地方在于测试用例实现的两个函数分别是SetUp()和TearDown(),这两个函数不是静态函数了。SetUp()函数是在一个测试用例的开始前执行。TearDown()函数是在一个测试用例的结束后执行。
- 代码:

#include <iostream> #include <gtest/gtest.h> using namespace std; class MyTestCase0 : public testing::Test
{
protected:
virtual void SetUp()
{
cout << "TestCase event0 : start" << endl;
} virtual void TearDown()
{
cout << "TestCase event0 : end" << endl;
}
}; class MyTestCase1 : public testing::Test
{
protected:
virtual void SetUp()
{
cout << "TestCase event1 : start" << endl;
}
virtual void TearDown()
{
cout << "TestCase event1 : end" << endl;
}
}; TEST_F(MyTestCase0, test0)
{
EXPECT_EQ(1, 1);
} TEST_F(MyTestCase0, test1)
{
EXPECT_EQ(1, 1);
} TEST_F(MyTestCase1, test0)
{
EXPECT_EQ(1, 1);
} TEST_F(MyTestCase1, test1)
{
EXPECT_EQ(1, 1);
} int main(int argc, char *argv[])
{
testing::InitGoogleTest(&argc, argv); return RUN_ALL_TESTS();
}
- 代码:
- 全局事件:需要创建一个自己的类,然后继承testing::Environment类,分别实现成员函数SetUp()和TearDown(),同时在main函数内进行调用,即"testing::AddGlobalTestEnvironment(new MyEnvironment);",通过调用函数我们可以添加多个全局的事件机制。需要注意的是,全局事件在--gtest_repeat参数大于1的情况下(可以通过--help查看其他参数),会执行多次
代码
- 上传到个人github中:https://github.com/longbozhan/sample/tree/master/gtest
参考
- https://www.shuzhiduo.com/A/n2d9gnDgJD/
- https://google.github.io/googletest/
C++单元测试框架gtest使用的更多相关文章
- Google单元测试框架gtest之官方sample笔记2--类型参数测试
gtest 提供了类型参数化测试方案,可以测试不同类型的数据接口,比如模板测试.可以定义参数类型列表,按照列表定义的类型,每个测试case都执行一遍. 本例中,定义了2种计算素数的类,一个是实时计算, ...
- Google单元测试框架gtest之官方sample笔记3--值参数化测试
1.7 sample7--接口测试 值参数不限定类型,也可以是类的引用,这就可以实现对类接口的测试,一个基类可以有多个继承类,那么可以测试不同的子类功能,但是只需要写一个测试用例,然后使用参数列表实现 ...
- Google单元测试框架gtest之官方sample笔记4--事件监控之内存泄漏测试
sample 10 使用event listener监控Water类的创建和销毁.在Water类中,有一个静态变量allocated,创建一次值加一,销毁一次值减一.为了实现这个功能,重载了new和d ...
- 简单易懂的单元测试框架-gtest(一)
简介 gtest是google开源的一个单元测试框架,以其简单易学的特点被广泛使用.该框架以第三方库的方式插入被测代码中.同其他单元测试框架相似,gtest也通过制作测试样例来进行代码测试.同 ...
- Google C++单元测试框架---Gtest框架简介(译文)
一.设置一个新的测试项目 在用google test写测试项目之前,需要先编译gtest到library库并将测试与其链接.我们为一些流行的构建系统提供了构建文件: msvc/ for Visual ...
- 简单易懂的单元测试框架-gtest(二)
简介 事件机制用于在案例运行前后添加一些操作(相当于挂钩函数).目前,gtest提供了三种等级的事件,分别: 全局级,所有案例执行的前后 TestSuite级,某一个案例集的前后 TestCa ...
- Google单元测试框架gtest之官方sample笔记1--简单用例
1.0 通用部分 和常见的测试工具一样,gtest提供了单体测试常见的工具和组件.比如判断各种类型的值相等,大于,小于等,管理多个测试的测试组如testsuit下辖testcase,为了方便处理初始化 ...
- Google C++单元测试框架GoogleTest(总)
之前一个月都在学习googletest框架,对googletest的文档都翻译了一遍,也都发在了之前的博客里,另外其实还有一部分的文档我没有发,就是GMock的CookBook部分:https://g ...
- Google C++单元测试框架GoogleTest---GTest的Sample1和编写单元测试的步骤
如果你还没有搭建gtest框架,可以参考我之前的博客:http://www.cnblogs.com/jycboy/p/6001153.html.. 1.The first sample: sample ...
随机推荐
- 9 — springboot整合jdbc、druid、druid实现日志监控 — 更新完毕
1.整合jdbc.druid 1).导入依赖 <dependency> <groupId>org.springframework.boot</groupId> &l ...
- CPU如何同时运行多个进程?
1 # -*- coding: utf-8 -*- 2 import re 3 mem = [x for x in re.split('[\r|\n]', ''' 4 store a 1 5 add ...
- flink03-----1.Task的划分 2.共享资源槽 3.flink的容错
1. Task的划分 在flink中,划分task的依据是发生shuffle(也叫redistrubute),或者是并行度发生变化 1. wordcount为例 package cn._51doit ...
- C++ 数字分类
1012 数字分类 (20分) 输入格式: 每个输入包含 1 个测试用例.每个测试用例先给出一个不超过 1000 的正整数 N,随后给出 N 个不超过 1000 的待分类的正整数.数字间 ...
- Linux学习 - 系统命令sudo权限
1 功能 root把超级用执行的命令赋予普通用户执行 2 使用 visudo 或 vim /etc/sudoers 说明: root 用户名 ALL=(ALL) 被管理主机的地址=(可使用的身份) A ...
- docker配置国内阿里云镜像源
使用docker默认镜像源下载镜像会很慢,因此很多情况下,我们在安装完docker以后都会修改为国内的镜像,这样在下载镜像的时候就不用等那么长时间了. 配置docker的镜像为阿里云镜像 方法一 $ ...
- zabbix之修改中文
#在zabbix服务器安装中文名包 root@ubuntu:~# sudo apt-get install language-pack-zh* #:修改环境变量 root@ubuntu:~# sudo ...
- 监控Linux服务器网站状态的SHELL脚本
1,监控httpd状态码的shell脚本代码. #!/bin/sh #site: www.jquerycn.cn # website[0]=www.jquerycn.cn/chuzu/' #网站1 m ...
- spring生成EntityManagerFactory的三种方式
spring生成EntityManagerFactory的三种方式 1.LocalEntityManagerFactoryBean只是简单环境中使用.它使用JPA PersistenceProvide ...
- Dubbo应用到web工程
一.创建提供者03-provider-web (1) 创建工程 创建Maven的web工程,然后创建实现类. (2) 导入依赖 Spring的版本为4.3.16 需要的依赖有: dubbo2.7.0版 ...