农民 发表于 2024-11-10 14:10:39

如何利用gtest编写C++单元测试代码

一.为什么要编写单元测试代码

相信许多人都不喜好编写单元测试代码,但是单元测试对我们来说真的很重要,单元测试可以暴暴露我们本身的代码的内部题目,从而保证我们本身的代码的正确性,避免出现各种bug。单元测试应该覆盖我们本身的代码的全部内部逻辑。
二.gtest是什么

gtest(全称googletest)是谷歌开源的一个单元测试框架,包含许多实用于具体场景的宏,可以方便的编写单元测试代码。
三.下载

源码下载地址:GitHub仓库。
源码下载完成之后构建工程,基本构建方法:
   cd googletest
mkdir build
cd build
cmake …
可以根据需要调解构建指令,这样就可以构建出符合的工程出来,然后编译生成需要的库文件就可以了。
四.利用方法

本文讨论三种场景,给出示例代码,但是不展示效果,可以自行测试,这样理解效果更好。
示例利用的是VS2022。
如果编译时报错检测到“RuntimeLibrary”的不匹配项: 值“MTd_StaticDebug”不匹配值“MDd_DynamicDebug”,缘故原由是编译gtest库时设置的编译参数和编写单元测试代码时不一样。可以在项目属性->C/C+±>代码生成->运行库,改为同等即可。
4.1 场景一

如果需要测试的多少个case互不相干,可以利用TEST宏测试。
TEST宏的第一个参数是测试套件名,第二个参数是测试案例名,是本身填写的。一个测试套件包含多少个测试案例。
示例代码:
//main.cpp

#include<gtest/gtest.h>

int Func(int data)
{
        return data;
}

TEST(TestDemo, test1)
{
        int data = 10;
        EXPECT_EQ(data, Func(10));
}

TEST(TestDemo, test2)
{
        int data = 5;
        EXPECT_EQ(data, Func(10));
}

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

        return RUN_ALL_TESTS();
}
4.2 场景二

如果需要测试的多少个case都具有相同的数据设置,可以利用TEST_F宏测试。
利用TEST_F宏需要先编写继承自testing::Test的固件类,用来提供数据设置。
TEST_F宏的第一个参数是测试套件名,必须要和固件类名同等,第二个参数是测试案例名,本身填写。一个测试套件包含多少个测试案例。
示例代码:
//demo.h

#ifndef DEMO_H
#define DEMO_H

class Demo
{
public:
        explicit Demo(int data) :m_data(data) {}

        ~Demo() noexcept = default;

        int GetData()
        {
                return m_data;
        }

private:
        int m_data;
};

#endif // !DEMO_H
//test.h

#ifndef TEST_H
#define TEST_H

#include<iostream>

#include<gtest/gtest.h>

#include"demo.h"

class DemoTest : public testing::Test
{
protected:
        DemoTest() :m_demo(10) {}

        ~DemoTest() noexcept override = default;

        void SetUp() override
        {
                //可以在这个函数里设置数据配置
                std::cout << "DemoTest::SetUp" << std::endl;
        }

        void TearDown() override
        {
                //可以在这个函数里释放资源
                std::cout << "DemoTest::TearDown" << std::endl;
        }

protected:
        Demo m_demo;
};

#endif // !TEST_H
//test.cpp

#include"test.h"

TEST_F(DemoTest, test1)
{
        int data = 10;
        EXPECT_EQ(data, m_demo.GetData());
}

TEST_F(DemoTest, test2)
{
        int data = 5;
        EXPECT_EQ(data, m_demo.GetData());
}
//main.cpp

#include<gtest/gtest.h>

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

        return RUN_ALL_TESTS();
}
4.3 场景三

如果同一个测试需要差别的输入参数,可以利用TEST_P宏测试。
利用TEST_P宏需要先编写继承自testing::TestWithParam<type>的固件类,<type>就是想要传入的参数的类型。
TEST_P宏的第一个参数是测试套件名,必须要和固件类名同等,第二个参数是测试案例名,本身填写。一个测试套件包含多少个测试案例。
示例代码:
//test.h

#ifndef TEST_H
#define TEST_H

#include<iostream>

#include<gtest/gtest.h>

class DemoTest : public testing::TestWithParam<int>
{
protected:
        DemoTest() = default;

        ~DemoTest() noexcept override = default;

        void SetUp() override
        {
                //可以在这个函数里设置数据配置
                std::cout << "DemoTest::SetUp" << std::endl;
        }

        void TearDown() override
        {
                //可以在这个函数里释放资源
                std::cout << "DemoTest::TearDown" << std::endl;
        }
};

#endif // !TEST_H
//test.cpp

#include"test.h"

TEST_P(DemoTest, test1)
{
        int data = 5;
        EXPECT_EQ(data, GetParam());
}

TEST_P(DemoTest, test2)
{
        int data = 10;
        EXPECT_EQ(data, GetParam());
}

INSTANTIATE_TEST_SUITE_P(DEMO, DemoTest, testing::Values(0, 5, 10));
//main.cpp

#include<gtest/gtest.h>

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

        return RUN_ALL_TESTS();
}
五.其他


[*] Linux下利用方法类似。
[*] 可以在官方文档查看更多用法。
[*] 可以利用gmock进行打桩。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: 如何利用gtest编写C++单元测试代码