Check—强大的C语言单元测试框架

                       

1. 前言

在看基数树源码时,发现源码里面有个deps的依赖文件夹,deps里是一个叫Check的源码安装包,Google之后发现Check原来是C语言单元测试框架。关于单元测试,Wikipedia的介绍点这里。

 

Check 最主要的优点是对于每一个测试用例的运行都 fork 一个子进程,这么做的原因是因为 C 语言的独特性:
  (1) 其它语言如 Java,Python,Ruby等,单元测试出错最多不过是抛出异常;
  (2) C 语言如果指针操作错误,乱指一气,可是会 coredump的。测试框架因此直接退出,用户是看不到任何返回的,只有郁闷的 coredump;
  (3) Check 的单元测试运行在 fork 的子进程中,可以避免测试框架由于 coredump 而崩溃。

网上介绍Check的使用基本都是同一例子,但如何使用没有说明,下面就详细介绍下如何利用Check进行单元测试。


2. Check简单使用

在使用Check前要安装Check,可以直接在终端输入:sudo apt-get install check进行安装,也可以使用源码包安装,源码具体安装方法可以参考官网。

我们现在也是要做一个只做加法的工程,整个工程的文件目录如下图所示,具体代码参考Github。工程文件夹里的TestAdd.c文件是将整个工程都放到了一个文件中,可以直接在终端运行:gcc TestAdd.c -lcheck进行编译,然后运行./test


结构图

TestAdd.c文件

    #include     #include "check.h"    #include     int add(int x1, int x2) {        return 0;    }    START_TEST(test_add) {        fail_unless(add(2, 3) == 5, "error, 2 + 3 != 5");    }    END_TEST    Suite * make_add_suite(void) {        Suite *s = suite_create("add");        TCase *tc_add = tcase_create("add");        suite_add_tcase(s, tc_add);        tcase_add_test(tc_add, test_add);        return s;    }    int main(void) {        int n;        SRunner *sr;        sr = srunner_create(make_add_suite());        srunner_run_all(sr, CK_NORMAL);        n = srunner_ntests_failed(sr);        srunner_free(sr);        return (n == 0) ? EXIT_SUCCESS : EXIT_FAILURE;    }
    
    
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30

下面介绍下工程中的文件夹功能,add目录下是功能函数,include中是头文件,测试文件在unit_test文件夹,makefile是整个项目的make文件。

makefile文件

    vpath %.h include  #vpath 指定搜索路径    vpath %.c add    vpath %.c unit_test    objects = add.o test_add.o    test: test_main.c $(objects)        gcc -I include $^ -o test -lcheck     all: $(objects)    $(objects): %.o: %.c        gcc -c -I include $< -o $@    .PHONY: clean    clean:        rm *.o test
    
    
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

add.c文件

    #include "add.h"    int add(int x1, int x2) {        return 0;    }
    
    
    
    
  • 1
  • 2
  • 3
  • 4

add.h文件

    #ifndef _ADD_H    #define _ADD_H    int add(int x1, int x2);    #endif
    
    
    
    
  • 1
  • 2
  • 3
  • 4

uni_test.h文件

    #ifndef _UNI_TEST_H    #define _UNI_TEST_H    #include "check.h"    Suite *make_add_suite(void);    #endif
    
    
    
    
  • 1
  • 2
  • 3
  • 4
  • 5

test_add.c文件

    #include "check.h"    #include "uni_test.h"    #include "add.h"    START_TEST(test_add) {        fail_unless(add(2, 3) == 5, "error, 2 + 3 != 5"); // "error, 2 + 3 != 5"是出错提示信息     }    END_TEST    Suite * make_add_suite(void) {        Suite *s = suite_create("add");       // 建立Suite        TCase *tc_add = tcase_create("add");  // 建立测试用例集        suite_add_tcase(s, tc_add);           // 将测试用例加到Suite中        tcase_add_test(tc_add, test_add);     // 测试用例加到测试集中        return s;    }
    
    
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

test_main.c文件

    #include "uni_test.h"    #include     int main(void) {        int n;        SRunner *sr;        sr = srunner_create(make_add_suite()); // 将Suite加入到SRunner        srunner_run_all(sr, CK_NORMAL);        n = srunner_ntests_failed(sr);         // 运行所有测试用例        srunner_free(sr);        return (n == 0) ? EXIT_SUCCESS : EXIT_FAILURE;    }
    
    
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

make test 可以编译生成可执行文件test./test 运行单元测试程序。


【完】

           

再分享一下我老师大神的人工智能教程吧。零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到我们人工智能的队伍中来!https://blog.csdn.net/jiangjunshow

你可能感兴趣的:(Check—强大的C语言单元测试框架)