【GTest学习】

news2024/12/28 8:13:44

1. GTest简介:

      GTest 就是 Google Test, 它是一个免费开源的测试框架, 用于编写测试用 C++语言编写的程序(C 程序也能用, 但是需要用 C++编译器编译)。gtest的官方网站是:http://code.google.com/p/googletest/

2.GTest下载与环境搭建:

GTest 下载地址如下。

http://code.google.com/p/googletest/

由于 Google 被 GFW 墙了,只能通过代理下载,或者从 GitHub 下载也行。

https://github.com/google/googletest

下载并解压之后,就可以看到gtest-1.7.0文件夹了,里面的内容如下图所示

   GTEST提供了对于多个不同平台的支持,例如msvc文件夹是用在微软Visual Studio中,xcode文件夹是用于Mac Xcode,codegrear文件夹是用于Borland C++ Builder,在Linux环境中,用的内容就是make文件夹了。为了保持程序的简洁,避免混淆试听,在这一步把所有不需要的文件全部删除,只保留需要的。其实只需要4个文件夹,如下图所示。其余的文件以及文件夹全部删除。

    打开make文件夹,会发现里面只有一个Makefile文件。查看Makefile文件内容,得知这是系统给出的编译samples文件夹中的第一个sample的命令。但是打开sample文件夹,又看到里面一大堆源文件。因此,可以打开samples文件夹,开始删文件,删到只剩下如图所示的三个文件为止。

   此时如果到make文件夹下,通过命令行执行 $ make && ./sample1_unittest 命令,可以看到测试的执行结果。不过如果打开Makefile查看一下,就会发现这个makefile只适用于编译sample1,如果我再增加一个被测 的源文件,又要重新写makefile,太麻烦了。于是,在这一步,我们改写一下Makefile。现在只剩下4个文件夹(include,make,samples,src),既然make里面的唯一一个文件也要被改写,那也没必要留 这个文件夹了。要做的第一件事情就是,把make文件夹,连同里面的Makefile文件全部删,然后,进入samples文件夹,自己创建一个文件,名为 Makefile,如图所示

然后,打开Makefile文件,写入以下内容,如图所示。这个新的Makefile是由刚才被我们删除的 Makefile改写而来的,里面涉及到一些makefile的语法和函数,注意下图中改写的Makefile第32行,我们编译的是后缀名为cpp的文件,而原来给的例子却以cc结尾。因此,你还要做一件事情,就是把 sample1.cc的文件名改为sample1.cpp,把sample1_unittest.cc的文件名改为 sample1_unittest.cpp,就大功告成了。

现在文件夹有三个(include,src,samples),被测的程序放在sample文件夹中。这个文件夹的名字看着也比较不方便,可以把它改为mycode,然后GTEST根目录的文件夹名称gtest-1.7.0也可以改为mygtest之类,用以满足我们的需求。如图所示:

现在,进入命令行进行编译执行操作: $ make && ./run_test,就可以看到结果了,如图所示:

假设我们现在有一个待测函数sqrt.cpp以及它的头文件sqrt.h,他们的内容如下:

(sqrt.cpp)

(sqrt.h)

(sqrt_unittest.cpp)

这个被测文件的作用是计算任意一个正整数的平方根,算法复杂度在log(n)级别。将以上三个文件放在mycode文件夹中,然后 make && ./run_test进行编译运行,就可以看到结果了:

总结:环境搭建完成之后,每次测试一个文件xxx.cpp以及它的xxx.h文件,就把这俩放入mycode文件夹,然后编写 xxx_unittest.cpp测试文件,也放进去。然后到这个目录下用命令行 make && ./run_test就可以了。要测试时,我们只需要三个文件放入mycode,然后命令行进入这个目录 make && ./run_test 即可完成测试。xxx_unittest.cpp这个文件名并不是固定,取成别的也无所谓,只是这样更容易辨 认哪一个文件是在测哪个函数而已。

3.Mock and Stub

        Mock通常是指,在测试一个对象A时,我们构造一些假的对象来模拟与A之间的交互,而这些Mock对象的行为是我们事先设定且符合预期。通过这些Mock对象来测试A在正常逻辑,异常逻辑或压力情况下工作是否正常。
引入Mock最大的优势在于:Mock的行为固定,它确保当你访问该Mock的某个方法时总是能够获得一个没有任何逻辑的直接就返回的预期结果。
Mock Object的使用通常会带来以下一些好处:
 隔绝其他模块出错引起本模块的测试错误。
 隔绝其他模块的开发状态,只要定义好接口,不用管他们开发有没有完成。
 一些速度较慢的操作,可以用Mock Object代替,快速返回。

在测试中,Mock 和Stub都是将被测代码和依赖代码之间的依赖被极大的隔离开来。那么这两者之间的区别是什么呢?
    在理解其区别之前,需要明白一点,他们都是为了同一个目标而出现的,代替依赖部分,让原先的“集成测试”简化为“单元测试”   mock:在程序代码中向被测试代码注入“依赖部分”,通过编写Mock代码的方式模拟出函数调用返回的结果。(注意:mock是注入,不是替换)
  stub:自己写代码代替“依赖部分”。它本身就是“依赖部分”的一个简化实现。(stub是 简单的替换)
     实际上,在能够使用mock的时候,就不应该选择使用stub。但是有时候是必须使用stub的,例如在对遗留代码进行测试时,该部分代码不支持“注入”,那么只能将“替代”这个过程外移,使用stub完成此任务了。

class DataGetter {

public:

...

bool Run();

...

private:

...

Client*  m_ptr_client;

...

};

.....

.....

bool DataGetter::Run() {

...

std::string data;

bool ret = m_ptr_client->GetData(data);

...

}

....

....

比如要对run这个函数进行单元测试,它内部调用了ptr_client->GetData(data)的方法,它是通过tcp协议从服务端取数据到data里,测试run这个函数,必然要构造data。

如果不“打桩”,要测试的话,我们就需要再从服务端去构造数据,而且还可能收到其他因素的影响。

原理:利用c++ virtual的特性,改变m_ptr_client指针所指向的对象,重写一个“打桩”测试类。

当然,前提是GetData的定义本身是virtual的。

假设Client的定义如下

class Client { ...... public:   virtual bool GetData(std::string& data); ...... };

我们只需要重写Getdata的方法,并且当参数data被传进来时,我们可以返回特定的值。

这一套方法,google已经提供了很好的一套框架:gmock

下面介绍一下它的用法

#include"client.h"//被mock的类的头文件

#include <gmock gmock.h="">// gmock 的头文件

class MockClient :public Client { public:   MockClient():Client() {}   MOCK_METHOD1(GetData, bool(std::string&)); }</gmock>

这里使用了一个宏MOCK_METHOD1

原形是MOCK_METHOD#1(#2, #3(#4) )

#1表示被mock的函数参数个数,#2表示被mock的函数名称,#3表示被mock的函数返回值,#4表示被mock的函数参数列表

使用mock:

    首先,要先改变m_ptr_client指向的对象,对于private的变量(可以使用#define private public 把所有的private变成public去使用),然后测试的时候,直接将 m_ptr_client = new MockClient()即可,不过要记得释放它之前new的资源。

4.GTest用法

  1. /*被测函数*/
  2. int Add(int iNum1,int iNum2)
  3. {
  4.     return (iNum1 + iNum2);
  5. }
  6. /*测试代码*/
  7. TEST(AddTest,HandleNoneZeroInput)
  8. {
  9.     EXPECT_EQ(3,Add(1,2));
  10. }
  11. /*主函数,用于启动Google Test框架*/
  12. void main(int argc, char* argv[])
  13. {
  14.     testing::InitGoogleTest(&argc,argv);
  15.     RUN_ALL_TESTS();
  16.     getch();
  17.     return;
  18. }

       断言

Google Test支持两类断言 ASSERT和EXPECT:

ASSERT_*系列

当检查失败时,退出当前函数。

EXPECT_*系列

当检查失败时,继续向下执行。

主要用于检查布尔,数值型,字符串,返回成功或失败,异常,浮点,类型。

布尔值检查

Fatal assertion

Nonfatal assertion

Verifies

ASSERT_TRUE(condition);

EXPECT_TRUE(condition);

Condition is true

ASSERT_FALSE(condition);

EXPECT_FALSE(condition);

Condition is false

数值型数据检查

Fatal assertion

Nonfatal assertion

Verifies

ASSERT_EQ(expected, actual)

EXPECT_EQ(expected, actual);

Expected ==actual

ASSERT_NE(val1, val2);

EXPECT_NE(val1, val2);

Val1 != val2

ASSERT_LT(val1,val2);

EXPECT_LT(val1,val2);

val1<val2

ASSERT_LE(val1,val2);

EXPECT_LE(val1,val2);

val1<=val2

ASSERT_GT(val1,val2);

EXPECT_GT(val1,val2);

val1>val2

ASSERT_GE(val1,val2);

EXPECT_GE(val1,val2);

val1>=val2

字符串检查

Fatal assertion

Nonfatal assertion

Verifies

ASSERT_STREQ(expected_str,actual_str);

EXPECT_STREQ(expected_str,actual_str);

the two C strings have the same content

ASSERT_STRNE(str1,str2);

EXPECT_STRNE(str1,str2);

the two C strings have different content

ASSERT_STRCASEEQ(expected_str,actual_str);

EXPECT_STRCASEEQ(expected_str,actual_str);

the two C strings have the same content, ignoring case

ASSERT_STRCASENE(str1,str2);

EXPECT_STRCASENE(str1,str2);

the two C strings have different content, ignoring case

  1.  Predicate Assertions

在使用EXPECT_TRUE或ASSERT_TRUE时,有时希望能够输出更加详细的信息,比如检查一个函数的返回值TRUE还是FALSE时,希望能够输出传入的参数是什么,以便失败后好跟踪。因此提供了如下谓词断言:

Fatal assertion

Nonfatal assertion

Verifies

ASSERT_PRED1(pred1, val1);

EXPECT_PRED1(pred1, val1);

pred1(val1)returns true

ASSERT_PRED2(pred2, val1, val2);

EXPECT_PRED2(pred2, val1, val2);

pred2(val1, val2)returns true

...

...

...

例:

bool MutuallyPrime(int m, int n)
{
    return Foo(m , n) > 1;
}

TEST(PredicateAssertionTest, Demo)
{
    int m = 5, n = 6;
    EXPECT_PRED2(MutuallyPrime, m, n);
}

ASSERT_PRED后面的数字表示pred函数有几个参数,目前支持的参数数量为<=5。

  1. 参数化

在设计测试案例时,经常需要考虑给被测函数传入不同的值的情况。我们之前的做法通常是写一个通用方法,然后编写在测试案例调用它。即使使用了通用方法,这样的工作也是有很多重复性的,程序员都懒,都希望能够少写代码,多复用代码。Google的程序员也一样,他们考虑到了这个问题,并且提供了一个灵活的参数化测试的方案。

例:如果要测一个函数用来判断一个数字是否是质数,我们可能需要测一组数据:

TEST(IsPrimeTest,HandleTrueReturn)
{
  EXPECT_TRUE(IsPrime(
3));
  EXPECT_TRUE(IsPrime(
5));
  EXPECT_TRUE(IsPrime(
11));
  EXPECT_TRUE(IsPrime(
23));

     EXPECT_TRUE(IsPrime(17));
}

假如测试的数据有成百上千的时候,那么我们要是写成百上千行代码,这是非常恐怖的,而参数化功能就很好的解决了这个问题,通过使用Gtest的参数化功能,上面的例子变成了:

classIsPrimeParamTest:public::testing::TestWithParam<int>
{

};

TEST_P(IsPrimeParamTest,HandleTrueReturn)
{

int n=GetParam();
EXPECT_TRUE(IsPrime(n));
}

INSTANTIATE_TEST_CASE_P(TrueReturn,IsPrimeParamTest,testing::Values(3,5,11,23,17));

在INSTANTIATE_TEST_CASE_P中第一个参数是测试案例的前缀,可以任意取;

第二个参数是测试案例的名称,需要和之前定义的参数化的类的名称相同,如:IsPrimeParamTest

第三个参数是可以理解为参数生成器,上面的例子使用test::Values表示使用括号内的参数。

Google Test提供了一系列的参数生成的函数:

Range(begin,end[,step])

范围在begin~end之间,步长为step,不包括end

Values(v1,v2,…,vN)

V1,v2到vN的值

ValuesIn(Container)

ValuesIn(begin,end)

从一个C类型的数组或是STL容器,或是迭代器中取值

Bool()

取false和true两个值

  1. 事件机制

Gtest提供的多种事件机制,事件机制主要是为了了在执行案例之前或之后做一些操作。gtest的事件一共有3种:

全局的,所有案例执行前后。

TestSuite级别的,在某一批案例中第一个案例前,最后一个案例执行后。

TestCase级别的,每个TestCase前后。

Gtest提供的使用方法:

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

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

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

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

1. SetUpTestCase() 方法在第一个TestCase之前执行
2. TearDownTestCase() 方法在最后一个TestCase之后执行

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

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

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

  1. 死亡测试

死亡测试”名字比较恐怖,这里的“死亡”指的的是程序的崩溃。通常在测试过程中,我们需要考虑各种各样的输入,有的输入可能直接导致程序崩溃,这时我们就需要检查程序是否按照预期的方式挂掉,这也就是所谓的“死亡测试”。gtest的死亡测试能做到在一个安全的环境下执行崩溃的测试案例,同时又对崩溃结果进行验证。Gtest提供了一种安全的测试环境,它可以在不影响其他案例执行的情况下,对崩溃案例的结果进行检查。

ASSERT_DEATH(statement,regex)

EXPECT_DETH(statement,regex)

Statement crashes with the given error

ASSERT_EXIT(statement,predicate,regex)

EXPECT_EXIT(statement,predicate,regex)

Statement exits with the given error and its exit code matches predicate

例子:

Void Foo()
{

int*pInt=0;
*pInt=
42;
}

TEST(FooDeathTest,Demo)
{
EXPECT_DEATH(Foo(),
"");
}

  1. 结果输出

Gtest有两种结果输出方式,一种是直接通过控制台输出结果,另一种方式是通过xml输出结果。

结果输出控制参数:

命令行参数

说明

--gtest_color=(yes|no|auto)

输出命令行时是否使用一些五颜六色的颜色。默认是auto。

--gtest_print_time

输出命令行时是否打印每个测试案例的执行时间。默认是不打印的。

--gtest_output=xml[:DIRECTORY_PATH\|:FILE_PATH]

将测试结果输出到一个xml中。

1.--gtest_output=xml: 不指定输出路径时,默认为案例当前路径。

2.--gtest_output=xml:d:\ 指定输出到某个目录

3.--gtest_output=xml:d:\foo.xml 指定输出到d:\foo.xml

如果不是指定了特定的文件路径,gtest每次输出的报告不会覆盖,而会以数字后缀的方式创建。

5.方法总结:

1.将被测试Code放在Dummy.cc文件中,

2.将需要做成的函数放在Stub.cc文件中

3.测试Case的SourceCode放在Dummy_unittest.cc文件中

4.编译SourceCode

过终gtest文件下的make目入如下的命令

make coverage=yes

5.执测试测

编译后,会在make目下生成可行程序Dummy_unittest

端中./Dummy_unittest 运行可行程序

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

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

相关文章

【雕爷学编程】Arduino动手做(195)---HT16k33 矩阵 8*8点阵屏模块4

37款传感器与模块的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&#x…

MFC计算分贝

分贝的一种定义是&#xff0c;表示功率量之比的一种单位&#xff0c;等于功率强度之比的常用对数的10倍&#xff1b; 主要用于度量声音强度&#xff0c;常用dB表示&#xff1b; 其计算&#xff0c;摘录网上一段资料&#xff1b; 声音的分贝值可以通过以下公式计算&#xff1…

python爬虫(七)_urllib2:urlerror和httperror

python爬虫(七)_urllib2&#xff1a;urlerror和httperror urllib2的异常错误处理 在我们用urlopen或opener.open方法发出一个请求时&#xff0c;如果urlopen或opener.open不能处理这个response&#xff0c;就产生错误。 这里主要说的是URLError和HTTPError,以及对它们的错误…

Vue Router 的query和params的区别?

区别一&#xff1a; &#xff08;1&#xff09;query相当于get请求&#xff0c;页面跳转的时候可以在地址栏看到请求参数 &#xff08;2&#xff09;params相当于post请求&#xff0c;参数不会在地址栏中显示&#xff0c;所以用params传值相对安全 &#xff08;简记&#xff1…

架构训练营学习笔记:5-1 计算架构模式之多级缓存架构

序 本节主要是计算架构。 多级缓存架构 缓存与缓冲&#xff1a;通常场景是读缓存&#xff0c;写缓冲。 缓存技术的本质&#xff1a;空间换时间&#xff0c;因此缓存架构属于高性能计算 架构。 缓存设计框架 主要考虑存什么&#xff1f;存多久&#xff1f;存哪里&#xff1f;如…

数字图像处理 --- 相机的内参与外参(CV学习笔记)

Pinhole Camera Model&#xff08;针孔相机模型&#xff09; 针孔相机是一种没有镜头、只有一个小光圈的简单相机。 光线穿过光圈并在相机的另一侧呈现倒立的图像。为了建模方便&#xff0c;我们可以把物理成像平面(image plane)上的图像移到实际场景(3D object)和焦点(focal p…

leetcode357周赛

2810. 故障键盘 核心思想&#xff1a;自己想的笨办法&#xff0c;枚举s&#xff0c;然后遇到i就翻转。比较好的方法就是双端队列&#xff0c;遇到i字母原本往后加的就往前加&#xff0c;然后读的时候反过来读&#xff0c;往前加的就往后加&#xff0c;读的话就从前往后&#x…

Java并发系列之八:ThreadPoolExecutor

线程池的意义 在讲解线程池之前&#xff0c;有些读者可能存在这样的疑惑&#xff1a;为什么需要线程池&#xff0c;线程池有什么优越性&#xff1f; 关于这个问题&#xff0c;主要从两个角度来进行解答: 减少开销 在大部分JVM上&#xff0c;用户线程与操作系统内核线程是1:1…

【论文阅读】对抗溯源图主机入侵检测系统的模仿攻击(NDSS-2023)

作者&#xff1a;伊利诺伊大学芝加哥分校-Akul Goyal、Gang Wang、Adam Bates&#xff1b;维克森林大学-Xueyuan Han、 引用&#xff1a;Goyal A, Han X, Wang G, et al. Sometimes, You Aren’t What You Do: Mimicry Attacks against Provenance Graph Host Intrusion Detect…

第一百二十三天学习记录:C++提高:STL-vector容器(下)(黑马教学视频)

vector插入和删除 功能描述&#xff1a; 对vector容器进行插入、删除操作 函数原型&#xff1a; push_back(ele); //尾部插入元素ele pop_back(); //删除最后一个元素 insert(const_iterator pos, ele); //迭代器指向位置pos插入元素ele insert(const_iterator pos, int cou…

Arduino 项目笔记 | Arduino LED Memory Game 颜色记忆游戏机

成果展示 颜色记忆游戏机 &#xff5c; Arduino DIY 1. 线路链连接 1.1 原理图 1.2 PCB 免费PCB打样 Arduino LED Memory Game 颜色记忆机资料下载 1.3 烧录 Bootloader 第二部分&#xff1a;Burn bootloader 2. 程序实现 #define NOTE_B0 31 #define NOTE_C1 33 #define NOT…

在Linux上进行项目部署--手动和自动

在Linux上进行项目部署–手动和自动 文章目录 在Linux上进行项目部署--手动和自动1、手动部署项目2、通过Shell脚本自动部署项目 1、手动部署项目 1、在IDEA中开发SpringBoot项目并打成jar包 在idea中的Maven中的package&#xff08;基于Springboot项目&#xff09; 2、将jar包…

React Native连接Zebra斑马打印机通过发送CPCL指令打印(Android 和 iOS通用)

自 2015 年发布以来&#xff0c;React Native 已成为用于构建数千个移动应用程序的流行跨平台移动开发框架之一。通常&#xff0c;我们有开发人员询问如何将 Link-OS SDK 与 React Native 应用程序集成&#xff0c;以便在 Zebra 打印机上打印标签。在本教程中&#xff0c;我们将…

机器视觉赛道持续火热,深眸科技坚持工业AI视觉切入更多应用领域

随着深度学习等算法的突破、算力的不断提升以及海量数据的持续积累&#xff0c;人工智能逐渐从学术界向工业界落地。而机器视觉作为人工智能领域中一个正在快速发展的分支&#xff0c;广泛应用于工业制造的识别、检测、测量、定位等场景&#xff0c;相较于人眼&#xff0c;在精…

系统架构设计高级技能 · 软件可靠性分析与设计(三)【系统架构设计师】

系列文章目录 系统架构设计高级技能 软件架构概念、架构风格、ABSD、架构复用、DSSA&#xff08;一&#xff09;【系统架构设计师】 系统架构设计高级技能 系统质量属性与架构评估&#xff08;二&#xff09;【系统架构设计师】 系统架构设计高级技能 软件可靠性分析与设计…

记录:在 TransnormerLLM 的纯线性注意力(改)中,出现的值异常现象

实验记录 注意&#xff0c;我为了让线性注意力在 fp16-mix 中稳定训练&#xff0c;作为以下修改。 输入线性注意力前&#xff0c;q 和 k 均做了以下操作 q q / torch.norm(q, dim-1, keepdimTrue) k k / torch.norm(k, dim-1, keepdimTrue)把 SRmsNorm 替换为普通的 RmsNo…

QT6 QML CMake工程添加qml到qrc中

参考QT官方文档 前言:使用qt6.2.4 qml的CMake工程时遇到qrc中的qml文件和图片无法使用的情况,查了好久终于找到解决办法,在此记录一下 1. 新建qml.qrc资源文件 可以在目录下看到qrc文件表示新建成功 2.给qrc文件添加前缀 添加完成后如下: 3. 修改CMakeLists.txt 我需要在一个q…

『SpringBoot 源码分析』自动配置

『SpringBoot 源码分析』自动装配 基于 2.2.9.RELEASE问题&#xff1a;Spring Boot 到底是如何进行自动配置的&#xff0c;都把哪些组件进行了自动配置&#xff1f; 首先创建测试主程序 package com.lagou;SpringBootApplication//标注在类上说明这个类是SpringBoot的主配置…

PHP序列化,反序列化

一.什么是序列化和反序列化 php类与对象 类是定义一系列属性和操作的模板&#xff0c;而对象&#xff0c;就是把属性进行实例化&#xff0c;完事交给类里面的方法&#xff0c;进行处理。 <?php class people{//定义类属性&#xff08;类似变量&#xff09;,public 代表可…

【计算机网络笔记】第一章

1、计算机网络定义 计算机网络主要是由一些通用的、可编程的硬件&#xff08;包含CPU、计算机、手机、智能电器…&#xff09;互连而成的&#xff0c;而这些硬件并非专门用来实现某一特定目的&#xff08;例如&#xff0c;传送数据或视频信号&#xff09;。这些可编程的硬件能…