GTest:TYPE_TESTING 全局函数时需要夹具?

GTest: fixture required when TYPE_TESTING global functions?

我想使用 TYPED_TEST 对一些全局模板函数进行单元测试。以下代码有效,我只是想知道是否有办法摆脱测试夹具,因为它似乎不需要..

#include <gtest/gtest.h>
#include <base/mathfunctions.h>

template <class T>
class MinTest : public testing::Test {};

// The list of types we want to test.
typedef ::testing::Types<int, float> Implementations;

TYPED_TEST_CASE(MinTest, Implementations);

TYPED_TEST(MinTest, ReturnsMinimumValue) 
{
   EXPECT_EQ(Base::Min<TypeParam>(-5, 5), -5);
}

... as it does not seem to be needed

这是需要的。查看单元测试源的独立简化 (把名字ReturnsMinimumValue搞得一团糟,不过没关系):

gtester.cpp

#include <gtest/gtest.h>

template<typename T>
struct foo {
    static T bar() {
        return 0;
    }
};

template <class T>
class MinTest : public testing::Test {};

typedef ::testing::Types<int, float> Implementations;

TYPED_TEST_CASE(MinTest, Implementations);

TYPED_TEST(MinTest, ReturnsMinimumValue)
{
    EXPECT_EQ(foo<TypeParam>::bar(),0);
}
int main(int argc, char **argv) {
    ::testing::InitGoogleTest(&argc, argv);
    return RUN_ALL_TESTS();
}

编译,link和运行即:

$ g++ -Wall -Wextra -o gtester gtester.cpp -lgtest -pthread && ./gtester
[==========] Running 2 tests from 2 test cases.
[----------] Global test environment set-up.
[----------] 1 test from MinTest/0, where TypeParam = int
[ RUN      ] MinTest/0.ReturnsMinimumValue
[       OK ] MinTest/0.ReturnsMinimumValue (0 ms)
[----------] 1 test from MinTest/0 (0 ms total)

[----------] 1 test from MinTest/1, where TypeParam = float
[ RUN      ] MinTest/1.ReturnsMinimumValue
[       OK ] MinTest/1.ReturnsMinimumValue (0 ms)
[----------] 1 test from MinTest/1 (0 ms total)

[----------] Global test environment tear-down
[==========] 2 tests from 2 test cases ran. (0 ms total)
[  PASSED  ] 2 tests.

现在让我们预处理源文件,并对预处理后的输出进行美化(因为它不漂亮):

$ g++ -E -P gtester.cpp | clang-format > gtester.ii

查看输出的最后 50 多行:

gtester.ii

...
...
int RUN_ALL_TESTS() __attribute__((warn_unused_result));
inline int RUN_ALL_TESTS() { return ::testing::UnitTest::GetInstance()->Run(); }
template <typename T> struct foo {
  static T bar() { return 0; }
};
template <class T> class MinTest : public testing::Test {};
typedef ::testing::Types<int, float> Implementations;
typedef ::testing::internal::TypeList<Implementations>::type
    gtest_type_params_MinTest_;
template <typename gtest_TypeParam_>
class MinTest_ReturnsMinimumValue_Test : public MinTest<gtest_TypeParam_> {
private:
  typedef MinTest<gtest_TypeParam_> TestFixture;
  typedef gtest_TypeParam_ TypeParam;
  virtual void TestBody();
};
bool gtest_MinTest_ReturnsMinimumValue_registered_ __attribute__((unused)) =
    ::testing::internal::TypeParameterizedTest<
        MinTest,
        ::testing::internal::TemplateSel<MinTest_ReturnsMinimumValue_Test>,
        gtest_type_params_MinTest_>::Register("",
                                              ::testing::internal::CodeLocation(
                                                  "gtester.cpp", 17),
                                              "MinTest", "ReturnsMinimumValue",
                                              0);
template <typename gtest_TypeParam_>
void MinTest_ReturnsMinimumValue_Test<gtest_TypeParam_>::TestBody() {
  switch (0)
  case 0:
  default:
    if (const ::testing::AssertionResult gtest_ar =
            (::testing::internal::EqHelper<(
                 sizeof(::testing::internal::IsNullLiteralHelper(
                     foo<TypeParam>::bar())) ==
                 1)>::Compare("foo<TypeParam>::bar()", "0",
                              foo<TypeParam>::bar(), 0)))
      ;
    else
      ::testing::internal::AssertHelper(
          ::testing::TestPartResult::kNonFatalFailure, "gtester.cpp", 19,
          gtest_ar.failure_message()) = ::testing::Message();
}
int main(int argc, char **argv) {
  ::testing::InitGoogleTest(&argc, argv);
  return RUN_ALL_TESTS();
}

我们的:

TYPED_TEST(MinTest, ReturnsMinimumValue)
{
    EXPECT_EQ(foo<TypeParam>::bar(),0);
}

扩展为以下定义:

template <typename gtest_TypeParam_>
class MinTest_ReturnsMinimumValue_Test;

从模板夹具派生:

template <class T> class MinTest;

来源于:

testing::Test;

MinTest_ReturnsMinimumValue_Test 从中继承:

virtual void TestBody();

并将其覆盖为:

void MinTest_ReturnsMinimumValue_Test<gtest_TypeParam_>::TestBody() {
  switch (0)
  case 0:
  default:
    if (const ::testing::AssertionResult gtest_ar =
            (::testing::internal::EqHelper<(
                 sizeof(::testing::internal::IsNullLiteralHelper(
                     foo<TypeParam>::bar())) ==
                 1)>::Compare("foo<TypeParam>::bar()", "0",
                              foo<TypeParam>::bar(), 0)))
      ;
    else
      ::testing::internal::AssertHelper(
          ::testing::TestPartResult::kNonFatalFailure, "gtester.cpp", 19,
          gtest_ar.failure_message()) = ::testing::Message();
}

它实现了扩展:

EXPECT_EQ(foo<TypeParam>::bar(),0);

不同的测试框架可能会完成类型参数化的测试用例 不同,但这就是 googletest 的工作方式。

多态class层次结构:

template <typename gtest_TypeParam_>
class MinTest_ReturnsMinimumValue_Test
    <-
template <class T> class MinTest
    <-
testing::Test

未生成,因为任何假定正在测试的方法调用 是多态的。 foo<TypeParam>::bar(),当然不是。多态的 class 生成层次结构是因为这是 googletest 的基本设计。