做单元测试,一般是验证函数输出数据与预期是否相符。但是也有例外,我们可能需要验证函数在接收到非法参数时是否会断言或抛异常,例如下面这个例子:
int foo(int *p)
{
assert(p != nullptr);
return *p;
}
当传给foo
的实参为nullptr
时,运行的预期结果就是断言后进程异常退出。这在单元测试中如何验证呢?如何不让这些验证异常值的用例干扰单元测试的工作流呢?
googletest提供了三组宏分别用于测试导致进程崩溃、导致进程退出、抛异常的函数,他们分别是EXPECT_DEATH
、EXPECT_EXIT
、EXPECT_THROW
及其各自的ASSERT版本。使用这些宏测试相应的函数不会干扰单元测试的工作流,换言之,函数断言并不会导致单元测试进程异常退出。而如果使用一般的宏,例如EXPECT_EQ
,来验证传入异常值时函数是否会断言时,单元测试进程也会异常退出,无法执行后续的测试用例。
以使用EXPECT_DEATH
宏测试foo
函数为例,测试用例需要这么写:
TEST(foo, death)
{
EXPECT_DEATH(foo(nullptr), ".*");
}
EXPECT_DEATH
,接受两个参数,分别是要测试的语句和要匹配的错误信息的正则表达式,这里我们并不关心实际的错误信息,所以直接传入".*"匹配任意信息。如果错误信息匹配不上的话,这个测试用例还是会被认为失败。
EXPECT_DEATH
的基本原理就是在执行测试语句前fork
一个进程出来执行,这样这个进程的死活就不会干扰单元测试进程本体了。
宏的手册以及更详细的原理,请查看googletest官方文档。