1

实施手动单元测试的架构是什么?如果要根据执行或失败的路径相应地返回值,我如何让这些返回的值有效地解释它?我要求用 C 语言给出相关的答案。

4

2 回答 2

4

我建议您考虑 C 中的单元测试,就像它在任何 xUnit 框架中的工作方式一样。

在 xUnit 中,测试被分组为“套件”(每个要测试的类一个套件是常见的),并且每个函数有一个或多个测试。在运行套件中的测试之前调用 setup 函数,并在该套件中的所有测试完成后调用 teardown 函数。

由于您的语言没有类,因此您可以将这些函数指针存储在函数指针结构中,并且可以遍历它们的数组,进行设置、测试和拆卸。

如果你愿意,你当然可以引入一个单元测试框架。C语言有很多可用的。您可以在Wikipedia上查看列表。

如果您不想变得大而正式,这里有一个完全用 C 实现的简单单元测试框架的示例,以及一个带有一些示例单元测试的示例“添加”应用程序。其中一项测试旨在故意失败,因此您可以看到失败的测试如何响应。最后,有一个测试执行应用程序的示例。单元测试框架不一定要大而重量级才能非常有效。


这两个模块是整个测试框架。

测试框架.h

/* testFramework.h */
#pragma once

typedef struct 
{
  int(*test)();
} UNITTEST;

typedef struct
{
  void(*setup)();
  void(*teardown)();
  UNITTEST *tests;
} UNITTESTSUITE;

extern int callTest(UNITTEST *test);
extern int callTestSuite(UNITTESTSUITE suite);
extern void testFailed(const char* testName, const int testLine, const char* message);

#define TESTFAILED(X) testFailed(__FILE__, __LINE__, X)

测试框架.c

/* testFramework.c */

#include <stdio.h>
#include <stdlib.h>
#include "testFramework.h"

int callTest(UNITTEST *test)
{
  return (*test).test();
};

int callTestSuite(UNITTESTSUITE suite)
{
  int rc = 1;
  UNITTEST* test = suite.tests;

  if (suite.setup)
    (suite.setup());
  while (test->test != 0)
  {
    if (callTest(test++))
    {
      printf(".");
    }
    else
    {
      printf("#");
      rc = 0;
    }
  }
  if (suite.teardown)
    (suite.teardown());

  printf("\n");
  return rc;
}

void testFailed(const char* testName, const int testLine, const char* message)
{
  fprintf(stderr, "%s(%i): ERROR test failed, %s\n", testName, testLine, message);
};

这是我需要测试的应用程序代码。

加法.h

/* addition.h */
#pragma once

extern int add(int x, int y);

加法.c

/* addition.c */

#include <stdio.h>
#include <stdlib.h>

int add(int x, int y)
{
  return x+y;
};

这是证明我的应用程序有效的单元测试。

附加测试.h

/* additionTests.h */
#pragma once

#include "testFramework.h"

extern void setupAdd();
extern void teardownAdd();
extern int testAddPositives();
extern int testAddFaultyTest();
extern int testAddNegatives();
extern int testAddMixed();
extern int testAddZeroes();

extern UNITTEST additionTests[];
extern UNITTESTSUITE additionSuite;

附加测试.c

/* additionTests.c */

#include <stdio.h>
#include <stdlib.h>

#include "testFramework.h"
#include "addition.h"

void setupAdd()
{
  return;
};

void teardownAdd()
{
  return;
};

int testAddPositives()
{
  int x=2, y=3;
  int expected = 5;
  int actual;

  actual = add(x,y);
  if (actual != expected)
  {
    TESTFAILED("actual not equal expected");
    return 0;
  }
  return 1;
};

int testAddFaultyTest()
{
  int x=2, y=2;
  int expected = 5;
  int actual;

  actual = add(x,y);
  if (actual != expected)
  {
    TESTFAILED("actual not equal expected");
    return 0;
  }
  return 1;
};

int testAddNegatives()
{
  int x=-2, y=-3;
  int expected = -5;
  int actual;

  actual = add(x,y);
  if (actual != expected)
  {
    TESTFAILED("actual not equal expected");
    return 0;
  }
  return 1;
};

int testAddMixed()
{
  int x=2, y=-3;
  int expected = -1;
  int actual;

  actual = add(x,y);
  if (actual != expected)
  {
    TESTFAILED("actual not equal expected");
    return 0;
  }
  return 1;
};

int testAddZeroes()
{
  int x=0, y=0;
  int expected = 0;
  int actual;

  actual = add(x,y);
  if (actual != expected)
  {
    TESTFAILED("actual not equal expected");
    return 0;
  }
  return 1;
};

/* add each additional unit test to this array */
UNITTEST additionTests[] = {&testAddPositives, &testAddFaultyTest, &testAddNegatives, &testAddMixed, &testAddZeroes, 0 };
UNITTESTSUITE additionSuite = {&setupAdd, &teardownAdd, additionTests};

最后,这是我的测试框架应用程序运行器。

testMain.c

/* testMain.c */

#include <stdio.h>
#include <stdlib.h>

/* include the unit test .h files here */
#include "additionTests.h"

int main(int argc, char**argv)
{
  /* call each unit test suite here */
  if (callTestSuite(additionSuite))
    return 0;

  return 2;
};
于 2013-10-11T20:31:03.350 回答
0

只需围绕您的代码编写一些代码。您可以使用 C 标准库之类的东西assert。假设您有一个int foo(int)应该返回1input的函数2。你是手动单元测试:

#include <assert.h>
#include "foo.h"

int main() {
    assert(foo(2) == 1);
}

显然,您会希望比这个玩具示例更健壮。

于 2013-10-11T09:24:05.837 回答