文章目录
🍎什么是gtest
gtest单元测试是Google的一套用于编写
C++测试的框架
,可以运行在很多平台上(包括Linux、Mac OS X、Windows、Cygwin等等)。基于xUnit架构。支持很多好用的特性,包括自动识别测试、丰富的断言、断言自定义、死亡测试、非终止的失败、生成XML报告等等。
⭐gtest的优点
好的测试应该有下面的这些特点,我们看看gtest是如何满足要求的。
- 测试应该是
独立的
、可重复的
。一个测试的结果不应该作为另一个测试的前提。gtest中每个测试运行在独立的对象中。如果某个测试失败了,可以单独地调试它。 - 测试应该是有清晰的结构的。gtest的测试有很好的组织结构,易于维护。
- 测试应该是
可移植
和可复用
的。有很多代码是不依赖平台的,因此它们的测试也需要不依赖于平台。gtest可以在多种操作系统、多种编译器下工作,有很好的可移植性。 - 测试失败时,应该给出尽可能详尽的信息。gtest在遇到失败时并不停止接下来的测试,而且还可以选择使用非终止的失败来继续执行当前的测试。这样一次可以测试尽可能多的问题。
- 测试框架应该避免让开发者维护测试框架相关的东西。gtest可以自动识别定义的全部测试,你不需要一一列举它们。
- 测试应该够快。gtest在满足测试独立的前提下,允许你复用共享数据,它们只需创建一次。
- gtest采用的是
xUnit
架构,你会发现和JUnit
、PyUnit
很类似,所以上手非常快。
⭐下载以及安装gtest
下载:
git clone https://github.com/google/googletest.git
1、$ cd googletest
2、$ cmake .
3、$ make
注意:如果在make 过程中报错,可在CMakeLists.txt 中增加如下行,再执行下面的命令:
SET(CMAKE_CXX_FLAGS "-std=c++11")
,重新执行
cmake .
以及
make
然后在lib目录下会生成:libgmock.a libgmock_main.a libgtest.a libgtest_main.a
4、最后我们再sudo make install。
⭐gtest断言类型
ASSERT_*断言和EXPECT_*断言的区别:
当ASSERT断言失败时,退出
当前TEST
,但是可以继续执行
其他TEST
(在Google Test中,每个TEST都是相互独立的,这意味着一个测试的失败不会影响其他测试的执行。)
换句话说,ASSERT_类型的断言是致命的,如果它们失败,那么测试将会停止执行。这可以帮助你快速地定位错误,但同时也会影响到测试的覆盖范围。而EXPECT_类型的断言则是非致命的,即使它们失败,测试仍将继续执行,这可以让你获得更多的测试覆盖范围,但也可能导致测试过于宽松,因为它们无法确保测试的正确性。
因此,使用哪种类型的断言取决于你的具体需求。如果你想快速地定位错误并停止测试,那么使用ASSERT_;如果你更关心测试覆盖范围,并希望测试能够继续执行,那么使用EXPECT_。
ASSERT_XXX(val1,val2)中val1是期待值而val2是实际值(EXPECT_XXX同理),下面总结了一些ASSERT和EXPECT,但是还不完整可以在用到的时候自己再去查
⭐头文件和库
当我们
make install
以后gtest相关头文件已经安装到/usr/local/include/下了,所以在使用的时候直接写
#include "gtest/gtest.h"
就可以了,而对于库文件,由于
make install
的时候已经将gtest的相关库安装到/usr/local/lib下了,所以后续使用的话直接链接就好了不用加路径
- libgtest.a或libgtest.so:gtest库的静态或动态链接库文件,包含gtest的实现代码。
- libgtest_main.a或libgtest_main.so:gtest库的静态或动态链接库文件,包含gtest的
主函数实现和启动测试的代码
(当我们的测试文件中没有main函数的话就可以链接到这个库就可以运行了)。
🎂gtest的使用【官网例子】
如何运行TEST程序:
1、写main方法,其中调用RUN_ALL_TESTS函数即可。
int main(int argc, char **argv) {
testing::InitGoogleTest(&argc, argv);
return RUN_ALL_TESTS();
}
2、我们也可以不用写main函数,那就需要链接
gtest_main.a
这个静态库
⭐sample1
sample1.h
#ifndefGOOGLETEST_SAMPLES_SAMPLE1_H_#defineGOOGLETEST_SAMPLES_SAMPLE1_H_// Returns n! (the factorial of n). For negative n, n! is defined to be 1.intFactorial(int n);// Returns true if and only if n is a prime number.boolIsPrime(int n);#endif// GOOGLETEST_SAMPLES_SAMPLE1_H_
sample1.cc
#include"sample1.h"// Returns n! (the factorial of n). For negative n, n! is defined to be 1.intFactorial(int n){int result =1;for(int i =1; i <= n; i++){
result *= i;}return result;}// Returns true if and only if n is a prime number.boolIsPrime(int n){// Trivial case 1: small numbersif(n <=1)returnfalse;// Trivial case 2: even numbersif(n %2==0)return n ==2;// Now, we have that n is odd and n >= 3.// Try to divide n by every odd number i, starting from 3for(int i =3;; i +=2){// We only have to try i up to the square root of nif(i > n / i)break;// Now, we have i <= n/i < n.// If n is divisible by i, n is not prime.if(n % i ==0)returnfalse;}// n has no integer factor in the range (1, n), and thus is prime.returntrue;}
sample_unittest.cc
#include<limits.h>#include"sample1.h"//已经make install到了/usr/local/include/gtest/gtest.h#include"gtest/gtest.h"namespace{//测试哪个函数以及这个测试的名字(也不一定要这样写,只是这样更加清楚)TEST(FactorialTest, Negative){// This test is named "Negative", and belongs to the "FactorialTest"// test case.// 断言,运行Factorial(-5)并对比结果是不是等于1(Factorial(-5)==1?)EXPECT_EQ(1,Factorial(-5));EXPECT_EQ(1,Factorial(-1));// 断言,运行Factorial(-10)并对比结果是不是小于0(0 > actorial(-10)?)EXPECT_GT(Factorial(-10),0);}TEST(FactorialTest, Zero){EXPECT_EQ(1,Factorial(0));}TEST(FactorialTest, Positive){EXPECT_EQ(1,Factorial(1));EXPECT_EQ(2,Factorial(2));EXPECT_EQ(6,Factorial(3));EXPECT_EQ(40320,Factorial(8));}// Tests IsPrime()TEST(IsPrimeTest, Negative){//预测是不是返回falseEXPECT_FALSE(IsPrime(-1));EXPECT_FALSE(IsPrime(-2));EXPECT_FALSE(IsPrime(INT_MIN));}TEST(IsPrimeTest, Trivial){EXPECT_FALSE(IsPrime(0));EXPECT_FALSE(IsPrime(1));EXPECT_TRUE(IsPrime(2));EXPECT_TRUE(IsPrime(3));}TEST(IsPrimeTest, Positive){EXPECT_FALSE(IsPrime(4));EXPECT_TRUE(IsPrime(5));EXPECT_FALSE(IsPrime(6));EXPECT_TRUE(IsPrime(23));}}// namespace
如何编译呢?👇
1、g++ sample1.cc sample1_unittest.cc -lgtest -std=c++14 -lgtest_main -lpthread -o test1
感兴趣的同学可以试试实现main方法来运行这个TEST文件
运行结果:
当我把第一个TEST进行改变
在这个sample里面用到
EXPECT_FALSE、EXPECT_TRUE、EXPECT_GT、EXPECT_EQ
⭐sample2
sample2.h
#ifndefGOOGLETEST_SAMPLES_SAMPLE2_H_#defineGOOGLETEST_SAMPLES_SAMPLE2_H_#include<string.h>// A simple string class.classMyString{private:constchar* c_string_;const MyString&operator=(const MyString& rhs);public:// Clones a 0-terminated C string, allocating memory using new.staticconstchar*CloneCString(constchar* a_c_string);//// C'tors// The default c'tor constructs a NULL string.MyString():c_string_(nullptr){}// Constructs a MyString by cloning a 0-terminated C string.explicitMyString(constchar* a_c_string):c_string_(nullptr){Set(a_c_string);}// Copy c'torMyString(const MyString& string):c_string_(nullptr){Set(string.c_string_);}//// D'tor. MyString is intended to be a final class, so the d'tor// doesn't need to be virtual.~MyString(){delete[] c_string_;}// Gets the 0-terminated C string this MyString object represents.constchar*c_string()const{return c_string_;}
size_t Length()const{return c_string_ ==nullptr?0:strlen(c_string_);}// Sets the 0-terminated C string this MyString object represents.voidSet(constchar* c_string);};#endif// GOOGLETEST_SAMPLES_SAMPLE2_H_
sample2.cc
#include"sample2.h"#include<string.h>// Clones a 0-terminated C string, allocating memory using new.constchar*MyString::CloneCString(constchar* a_c_string){if(a_c_string ==nullptr)returnnullptr;const size_t len =strlen(a_c_string);char*const clone =newchar[len +1];memcpy(clone, a_c_string, len +1);return clone;}// Sets the 0-terminated C string this MyString object// represents.voidMyString::Set(constchar* a_c_string){// Makes sure this works when c_string == c_string_constchar*const temp =MyString::CloneCString(a_c_string);delete[] c_string_;
c_string_ = temp;}
sample2_unittest.cc
#include"sample2.h"#include"gtest/gtest.h"namespace{// In this example, we test the MyString class (a simple string).// Tests the default c'tor.TEST(MyString, DefaultConstructor){const MyString s;EXPECT_STREQ(nullptr, s.c_string());EXPECT_EQ(0u, s.Length());}constchar kHelloString[]="Hello, world!";// Tests the c'tor that accepts a C string.TEST(MyString, ConstructorFromCString){const MyString s(kHelloString);EXPECT_EQ(0,strcmp(s.c_string(), kHelloString));EXPECT_EQ(sizeof(kHelloString)/sizeof(kHelloString[0])-1, s.Length());}// Tests the copy c'tor.TEST(MyString, CopyConstructor){const MyString s1(kHelloString);const MyString s2 = s1;EXPECT_EQ(0,strcmp(s2.c_string(), kHelloString));}// Tests the Set method.TEST(MyString, Set){
MyString s;
s.Set(kHelloString);EXPECT_EQ(0,strcmp(s.c_string(), kHelloString));// Set should work when the input pointer is the same as the one// already in the MyString object.
s.Set(s.c_string());EXPECT_EQ(0,strcmp(s.c_string(), kHelloString));// Can we set the MyString to NULL?
s.Set(nullptr);EXPECT_STREQ(nullptr, s.c_string());}}// namespace
如何编译呢:
g++ sample2.cc sample2_unittest.cc -lgtest -std=c++14 -lgtest_main -lpthread -o test2
运行结果:
在这个sample里面对比上一个sample用到了
EXPECT_STREQ(str1,str2)
断言,这是用来比较str1和str2的值是否相等,从这个例子上来说,gtest对类的测试也是完全支持的。
⭐sample3
sample3-inl.h
#ifndefGOOGLETEST_SAMPLES_SAMPLE3_INL_H_#defineGOOGLETEST_SAMPLES_SAMPLE3_INL_H_#include<stddef.h>// Queue is a simple queue implemented as a singled-linked list.//// The element type must support copy constructor.template<typenameE>// E is the element typeclassQueue;// QueueNode is a node in a Queue, which consists of an element of// type E and a pointer to the next node.template<typenameE>// E is the element typeclassQueueNode{friendclassQueue<E>;public:// Gets the element in this node.const E&element()const{return element_;}// Gets the next node in the queue.
QueueNode*next(){return next_;}const QueueNode*next()const{return next_;}private:// Creates a node with a given element value. The next pointer is// set to NULL.explicitQueueNode(const E& an_element):element_(an_element),next_(nullptr){}// We disable the default assignment operator and copy c'tor.const QueueNode&operator=(const QueueNode&);QueueNode(const QueueNode&);
E element_;
QueueNode* next_;};template<typenameE>// E is the element type.classQueue{public:// Creates an empty queue.Queue():head_(nullptr),last_(nullptr),size_(0){}// D'tor. Clears the queue.~Queue(){Clear();}// Clears the queue.voidClear(){if(size_ >0){// 1. Deletes every node.
QueueNode<E>* node = head_;
QueueNode<E>* next = node->next();for(;;){delete node;
node = next;if(node ==nullptr)break;
next = node->next();}// 2. Resets the member variables.
head_ = last_ =nullptr;
size_ =0;}}voidEnqueue(const E& element){
QueueNode<E>* new_node =newQueueNode<E>(element);if(size_ ==0){
head_ = last_ = new_node;
size_ =1;}else{
last_->next_ = new_node;
last_ = new_node;
size_++;}}// Removes the head of the queue and returns it. Returns NULL if// the queue is empty.
E*Dequeue(){if(size_ ==0){returnnullptr;}const QueueNode<E>*const old_head = head_;
head_ = head_->next_;
size_--;if(size_ ==0){
last_ =nullptr;}
E* element =newE(old_head->element());delete old_head;return element;}// Applies a function/functor on each element of the queue, and// returns the result in a new queue. The original queue is not// affected.template<typenameF>
Queue*Map(F function)const{
Queue* new_queue =newQueue();for(const QueueNode<E>* node = head_; node !=nullptr;
node = node->next_){
new_queue->Enqueue(function(node->element()));}return new_queue;}private:
QueueNode<E>* head_;// The first node of the queue.
QueueNode<E>* last_;// The last node of the queue.
size_t size_;// The number of elements in the queue.// We disallow copying a queue.Queue(const Queue&);const Queue&operator=(const Queue&);};#endif// GOOGLETEST_SAMPLES_SAMPLE3_INL_H_
sample3_unittest.cc
#include"sample3-inl.h"#include"gtest/gtest.h"namespace{// 测试类继承自 testing::Test,这样这个测试类可以反复使用classQueueTestSmpl3:public testing::Test{protected:// You should make the members protected s.t. they can be// accessed from sub-classes.// virtual void SetUp() will be called before each test is run. You// should define it if you need to initialize the variables.// Otherwise, this can be skipped.voidSetUp()override{
q1_.Enqueue(1);
q2_.Enqueue(2);
q2_.Enqueue(3);}// virtual void TearDown() will be called after each test is run.// You should define it if there is cleanup work to do. Otherwise,// you don't have to provide it.//// virtual void TearDown() {// }// A helper function that some test uses.staticintDouble(int n){return2* n;}// A helper function for testing Queue::Map().voidMapTester(const Queue<int>* q){// Creates a new queue, where each element is twice as big as the// corresponding one in q.const Queue<int>*const new_q = q->Map(Double);// Verifies that the new queue has the same size as q.ASSERT_EQ(q->Size(), new_q->Size());// Verifies the relationship between the elements of the two queues.for(const QueueNode<int>*n1 = q->Head(),*n2 = new_q->Head();
n1 !=nullptr; n1 = n1->next(), n2 = n2->next()){EXPECT_EQ(2* n1->element(), n2->element());}delete new_q;}// Declares the variables your tests want to use.
Queue<int> q0_;
Queue<int> q1_;
Queue<int> q2_;};// When you have a test fixture, you define a test using TEST_F// instead of TEST.// Tests the default c'tor.TEST_F(QueueTestSmpl3, DefaultConstructor){// You can access data in the test fixture here.EXPECT_EQ(0u, q0_.Size());}// Tests Dequeue().TEST_F(QueueTestSmpl3, Dequeue){int* n = q0_.Dequeue();EXPECT_TRUE(n ==nullptr);
n = q1_.Dequeue();ASSERT_TRUE(n !=nullptr);EXPECT_EQ(1,*n);EXPECT_EQ(0u, q1_.Size());delete n;
n = q2_.Dequeue();ASSERT_TRUE(n !=nullptr);EXPECT_EQ(2,*n);EXPECT_EQ(1u, q2_.Size());delete n;}// Tests the Queue::Map() function.TEST_F(QueueTestSmpl3, Map){MapTester(&q0_);MapTester(&q1_);MapTester(&q2_);}}// namespace
如何编译呢?👇
g++ sample3_unittest.cc -lgtest -std=c++14 -lgtest_main -lpthread -o test3
运行效果:
1、这个sample中用到了
测试类继承自testing::Test
,主要是为了下面两个原因:
- 设置共享的状态和资源:你可能需要创建一些共享的对象,在多个测试用例中
复用
它们。你可以在 SetUp() 函数中创建这些对象,然后在 TearDown() 函数中释放它们。这样可以保证这些对象在所有测试用例执行之前创建,并且在所有测试用例执行之后销毁。 - 提供公共的函数和工具函数:你可能需要在多个测试用例中使用相同的函数和工具函数。将这些函数和工具函数放在测试类中可以使其易于共享和重用。
2、继承自 testing::Test 的测试类可以定义
SetUp() 和 TearDown() 函数
。在
每个测试用例之前和之后
,Google Test 会
自动调用这些函数
,以帮助你设置和清理测试环境。
3、此外,如果你需要在测试用例中使用类的成员函数或变量,那么继承自 testing::Test 可以使这些成员函数和变量在测试用例中可见。
如果将测试类的成员变量设置为
protected
或
public
,测试方法就可以直接访问这些成员变量了。使用public或protected可以方便地访问测试类的成员变量和方法,但是会暴露内部实现细节。而使用private则可以更好地封装内部实现,但是需要提供一些公共的接口来进行测试。在实际应用中,需要根据实际情况来选择合适的访问控制方式。
4、不知道大家注意到没有,这里用到的宏是TEST_F而不是原来的TEST,在使用
测试夹具
(继承自testing::Test的测试类)的时候一般是使用TEST_F
①TEST宏和TEST_F宏都是gtest库中用于定义测试用例的宏,它们的主要区别在于测试用例的初始化和清理方式不同。TEST宏用于定义一个独立的测试用例,TEST_F宏用于定义一个测试夹具(fixture),它提供了一种在测试用例之间共享状态和代码的方法。TEST_F的第一个参数就不能乱取了,必须是测试夹具(继承自testing::Test的测试类),在测试用例运行前,会对测试夹具进行初始化操作(
SetUp()
),在运行时会通过第一个参数
创建一个测试夹具的实例
,在运行后,会对测试夹具进行清理(
TearDown()
)。
②TEST宏用于定义独立的测试用例,而TEST_F宏用于定义一个测试夹具,可以在测试用例之间共享状态和代码(
其实就是说一些初始条件不用反复设置代码复用性好,不自己单独创建实例,让你的测试代码更加易读和易维护。
)。选择使用哪种宏取决于你的测试需要。如果你的测试需要在运行之前进行一些初始化操作,或者需要在测试用例之间共享状态,那么使用TEST_F宏是更合适的选择。如果你的测试用例之间没有共享状态,那么使用TEST宏即可。
gtest的相关内容先讲到这,如果以后在开发过程中遇到其他高级用法或者是文章中没提到的方法,到时候再继续更新,感谢您看到这,希望这篇文章对您能有所帮助,谢谢!
版权归原作者 CAccept 所有, 如有侵权,请联系我们删除。