百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

别只会写代码,不会测试代码!教你Windows下如何搭建CUNIT环境?

nanshan 2024-11-01 12:40 9 浏览 0 评论

Windows下如何搭建CUNIT环境资料很多,但是错误不少或者讲解不清晰,很容易让人跌入坑中,现在介绍如下。

  1 安装mingw

  网上多处文章介绍下载mingw-get,由于现在网上大多数mingw-get均为32位的,所以不能用这个。所以大家需要下载https://sourceforge.net/projects/mingw-w64/files/下载mingw-w64-install.exe,在安装的时候注意:

  i686为32位的,x86_64为64位的。所以这里我们选择x86_64。安装完毕配置环境变量。加入MINGW_HOME环境变量,我这里路径为:C:\mingw-w64\x86_64-8.1.0-posix-seh-rt_v6-rev0\。在PATH中设置%MinGW_HOME%\mingw64\bin\和%MinGW_HOME%\mingw64\include\。打开cmd,输入gcc,如果返回gcc: fatal error: no input files则说明配置正确。

  2 下载CUNIT tar包

  CUNIT tar包是为Linux开发的,但是在Windows下可用msys2工具进行编译。我解压完毕放在C:\CUnit-2.1-3目录下

  3 安装msys2

  msys2可以让你在Windows下编译Linux的代码,目前网站上有简易版和完全版,简易版下载以后还需要安装各个命令的插件,比较麻烦,所以建议安装完全版,完整版的文件名为msys+7za+wget+svn+git+mercurial+cvs-rev13.7z(这个文件比较大,下载需要1个多小时)。解压完毕请把文件夹msys全部拷贝到mingw的目录下,我这里为C:\mingw-w64\x86_64-8.1.0-posix-seh-rt_v6-rev0\mingw64。

  进入C:\mingw-w64\x86_64-8.1.0-posix-seh-rt_v6-rev0\mingw64\msys,打开msys.bat。执行以下命令:

cd C:\CUnit-2.1-3 #解压的CUnit的根目录

  libtoolize

  automake --add-missing

  autoreconf

  ./configure --prefix=/mingw

  make

  make install

  安装完毕把C:\mingw-w64\x86_64-8.1.0-posix-seh-rt_v6-rev0\mingw64\msys\mingw\lib\ libcunit.a拷贝到C:\mingw-w64\x86_64-8.1.0-posix-seh-rt_v6-rev0\mingw64\lib。(不拷贝在下面gcc或者clang运行中,加入-lcunit参数会提示..lib: can’t find -lcunit的错误)

  接下来把C:\mingw-w64\x86_64-8.1.0-posix-seh-rt_v6-rev0\mingw64\msys\mingw\include\CUnit\目录中的所有.h文件拷贝到C:\mingw-w64\x86_64-8.1.0-posix-seh-rt_v6-rev0\mingw64\include中去。

  4 建立测试文件和被测文件

  在任意工程目录下建立被测试文件:process.h和process.c

  process.h:

extern int process(int x, int y, int z);

  process.c:

#include <stdio.h>


  int process(int x, int y, int z){

   int k=0;

   int j=0;

   if ((x>3) && (z<10)){

   k=x*y-1;

   j=k^2;

   }

   if((x==4) || (y>5)){

   j=x*j+10;

   }

   j=j%3;

   return k+j;

  }

  和测试文件test_main.c:

#include "process.h"


  /* test cases */


  void test_process_1(void)

  {

   CU_ASSERT(process(4,6,9)==24);

  }

  void test_process_2(void)

  {

   CU_ASSERT(process(4,5,10)==1);

  }

  void test_process_3(void)

  {

   CU_ASSERT(process(5,4,9)==21);

  }

  void test_process_4(void)

  {

   CU_ASSERT(process(2,5,10)==0);

  }

  CU_TestInfo tests[] = {

  {"test 1", test_process_1 },

  {"test 2", test_process_2 },

  {"test 3", test_process_3 },

  {"test 4", test_process_4 },

   CU_TEST_INFO_NULL

  };

  /* suite init */

  int suite_init(void)

  {

   return 0;

  }

  int suite_clean(void)

  {

   return 0;

  }

  void suite_setup(void)

  {

  }

  void suite_teardown(void)

  {

  }

  CU_SuiteInfo suites[] = {

   {"suite 1", suite_init, suite_clean, suite_setup, suite_teardown, tests},

   CU_SUITE_INFO_NULL

  };

  /* registry */


  int main(int argc, char* argv[])

  {

   CU_ErrorCode err;

  /* init */

  printf("init\n");

  err = CU_initialize_registry();

  if( err ){

  printf("CU_initialize_registry: %d\n", err);

  return err;

  }

  /* add suites and tests */

  printf("add suites and tests\n");

  err = CU_register_suites(suites);

  if( err )

  {

  printf("CU_register_suites: %d\n", err);

  }

  CU_pTestRegistry reg = CU_get_registry();

  printf("CU_get_registry: %d/%d/%u\n", reg->uiNumberOfSuites, reg->uiNumberOfTests, (long)reg->pSuite);

  /* run auto */

  printf("run auto\n");

  /**** Automated Mode *****************/

  CU_set_output_filename("TestProcess");

  CU_list_tests_to_file();

  CU_automated_run_tests();

  //************************************/

  printf("run basic\n");

  /***** Basice Mode *******************/

  CU_basic_set_mode(CU_BRM_VERBOSE);

  CU_basic_run_tests();

  //************************************/

  /*****Console Mode ********************

  CU_console_run_tests();

  //************************************/

  /* end */

  printf("end\n");

  CU_cleanup_registry();

  err = CU_get_error();

  if( err )

  {

   printf("error: %d", err);

   }

   return err;

  }

  通过以下命令编译(可以把这个命令放入一个bat文件中):

gcc process.c test_main.c -o test -I/mingw-w64/x86_64-8.1.0-posix-seh-rt_v6-rev0/mingw64/include -L/mingw-w64/x86_64-8.1.0-posix-seh-rt_v6-rev0/mingw64/lib -lcunit -static

  编译完成出现一个名为test.exe的文件,在cmd中运行它,会产生测试结果。

C:\MyC\process>test

  init

  add suites and tests

  CU_get_registry: 1/4/1909760

  run auto

  run basic


   CUnit - A unit testing framework for C - Version 2.1-3

   http://cunit.sourceforge.net/

  Suite: suite 1

   Test: test 1 ...passed

   Test: test 2 ...passed

   Test: test 3 ...passed

   Test: test 4 ...passed

  Run Summary: Type Total Ran Passed Failed Inactive

  suites 1 1 n/a 0 0

  tests 4 4 4 0 0

  asserts 4 4 4 0 n/a

  Elapsed time = 0.048 seconds

  end

  5. 使用VS Code IDE

  这里基本上就可以使用CUNIT进行工作了,但是为了调试的方便,我们需要安装一个IDE,这里建议使用微软的VS Code。

  5.1 下载并且安装Visual Studio Code

  5.2 下载并且安装LLVM。在选择三个单选框时选择 第二个选项 Add LLVM to the system PATH for all users。安装完毕在cmd窗口中输入clang,提示 no input file 表示安装成功。

  5.3 打开VS Code

  在插件中输入Chinese,安装重启VS Code变为中文版

  在插件中输入c/c++,选第一个安装

  在插件中输入code runner,选第一个安装

  在插件中输入clang,选第一个安装

  5.4 配置4个json配置文件

  刚才的文件是在C:\MyC\process目录下,在C:\MyC\.vscode下建立四个json文件。

  launch.json

// https://github.com/Microsoft/vscode-cpptools/blob/master/launch.md

  {

   "version": "0.2.0",

   "configurations": [

   {

   "name": "(gdb) Launch", // 配置名称,将会在启动配置的下拉菜单中显示

   "type": "cppdbg", // 配置类型,这里只能为cppdbg

   "request": "launch", // 请求配置类型,可以为launch(启动)或attach(附加)

   "program": "${fileDirname}/${fileBasenameNoExtension}.exe", // 将要进行调试的程序的路径

   "args": [], // 程序调试时传递给程序的命令行参数,一般设为空即可

   "stopAtEntry": true, // 设为true时程序将暂停在程序入口处,我一般设置为true

   "cwd": "${workspaceFolder}", // 调试程序时的工作目录

   "environment": [], // 环境变量

   "externalConsole": true, // 调试时是否显示控制台窗口,一般设置为true显示控制台

   "internalConsoleOptions": "neverOpen", // 如果不设为neverOpen,调试时会跳到“调试控制台”选项卡,你应该不需要对gdb手动输命令吧?

   "MIMode": "gdb", // 指定连接的调试器,可以为gdb或lldb。但我没试过lldb

   "miDebuggerPath": "gdb.exe", // 调试器路径,Windows下后缀不能省略,Linux下则不要

   "setupCommands": [ // 用处未知,模板如此

   {

   "description": "Enable pretty-printing for gdb",

   "text": "-enable-pretty-printing",

   "ignoreFailures": false

   }

   ],

   "preLaunchTask": "Compile" // 调试会话开始前执行的任务,一般为编译程序。与tasks.json的label相对应

   }

   ]

  }

  setting.json

{

   "files.defaultLanguage": "c", // ctrl+N新建文件后默认的语言

   "editor.formatOnType": true, // 输入时就进行格式化,默认触发字符较少,分号可以触发

   "editor.snippetSuggestions": "top", // snippets代码优先显示补全

   "code-runner.runInTerminal": true, // 设置成false会在“输出”中输出,无法输入

   "code-runner.executorMap": {

   "c": "cd $dir && clang $fileName -o $fileNameWithoutExt.exe -Wall -g -Og -static-libgcc -fcolor-diagnostics --target=x86_64-w64-mingw -std=c11 && $dir$fileNameWithoutExt",

   "cpp": "cd $dir && clang++ $fileName -o $fileNameWithoutExt.exe -Wall -g -Og -static-libgcc -fcolor-diagnostics --target=x86_64-w64-mingw -std=c++17 && $dir$fileNameWithoutExt"

  }, // 设置code runner的命令行

   "code-runner.saveFileBeforeRun": true, // run code前保存

   "code-runner.preserveFocus": true, // 若为false,run code后光标会聚焦到终端上。如果需要频繁输入数据可设为false

   "code-runner.clearPreviousOutput": false, // 每次run code前清空属于code runner的终端消息

   "C_Cpp.clang_format_sortIncludes": true, // 格式化时调整include的顺序(按字母排序)

   "C_Cpp.intelliSenseEngine": "Default", // 可以为Default或Tag Parser,后者较老,功能较简单。具体差别参考cpptools扩展文档

   "C_Cpp.errorSquiggles": "Disabled", // 因为有clang的lint,所以关掉

   "C_Cpp.autocomplete": "Disabled", // 因为有clang的补全,所以关掉

   "clang.cflags": [ // 控制c语言静态检测的参数

   "--target=x86_64-w64-mingw",

   "-std=c11",

   "-Wall",

   "-lcunit"

   ],

   "clang.cxxflags": [ // 控制c++静态检测时的参数

   "--target=x86_64-w64-mingw",

   "-std=c++17",

   "-Wall",

   "-lcunit"

   ],

   "clang.completion.enable":true // 效果效果比cpptools要好

   }

  task.json

// https://code.visualstudio.com/docs/editor/tasks

  {

   "version": "2.0.0",

   "tasks": [

   {

   "label": "Compile", // 任务名称,与launch.json的preLaunchTask相对应

   "command": "clang", // 要使用的编译器,C++用clang++

   "args": [

   "${file}",

   "-o", // 指定输出文件名,不加该参数则默认输出a.exe,Linux下默认a.out

   "${fileDirname}/${fileBasenameNoExtension}.exe",

   "-g", // 生成和调试有关的信息

   "-Wall", // 开启额外警告

   "-static-libgcc", // 静态链接libgcc

   "--target=x86_64-w64-mingw", // clang的默认target为msvc,不加这一条就会找不到头文件;Linux下去掉这一条

   "-std=c11" ,// C++最新标准为c++17,或根据自己的需要进行修改

   ], // 编译命令参数

   "type": "shell", // 可以为shell或process,前者相当于先打开shell再输入命令,后者是直接运行命令

   "group": {

   "kind": "build",

   "isDefault": true // 设为false可做到一个tasks.json配置多个编译指令,需要自己修改本文件,我这里不多提

   },

   "presentation": {

   "echo": true,

   "reveal": "always", // 在“终端”中显示编译信息的策略,可以为always,silent,never。具体参见VSC的文档

   "focus": false, // 设为true后可以使执行task时焦点聚集在终端,但对编译c和c++来说,设为true没有意义

   "panel": "shared" // 不同的文件的编译信息共享一个终端面板

   }

   // "problemMatcher":"$gcc" // 如果你不使用clang,去掉前面的注释符,并在上一条之后加个逗号。照着我的教程做的不需要改(也可以把这行删去)

   }

   ]

  }

c_cpp_properties.json

{

   "configurations": [

   {

   "name": "MinGW",

   "intelliSenseMode": "gcc-x64",

   "compilerPath": "C:/mingw-w64/x86_64-8.1.0-posix-seh-rt_v6-rev0/mingw64/bin/gcc.exe",

   "includePath": [

   "${workspaceFolder}"

  ],

   "defines": [],

   "browse": {

   "path": [

   "${workspaceFolder}"

   ],

   "limitSymbolsToIncludedHeaders": true,

   "databaseFilename": ""

   },

   "cStandard": "c11",

   "cppStandard": "c++17"

   }

   ],

   "version": 4

  }

  就可以使用这个工具运行C程序了。但是运行CUNIT程序请在CMD中运行。

请关注+私信回复:“测试”就可以免费拿到Python测试开发学习路线及软件测试资料包,快速让自己变强!

相关推荐

实战派 | Java项目中玩转Redis6.0客户端缓存

铺垫首先介绍一下今天要使用到的工具Lettuce,它是一个可伸缩线程安全的redis客户端。多个线程可以共享同一个RedisConnection,利用nio框架Netty来高效地管理多个连接。放眼望向...

轻松掌握redis缓存穿透、击穿、雪崩问题解决方案(20230529版)

1、缓存穿透所谓缓存穿透就是非法传输了一个在数据库中不存在的条件,导致查询redis和数据库中都没有,并且有大量的请求进来,就会导致对数据库产生压力,解决这一问题的方法如下:1、使用空缓存解决对查询到...

Redis与本地缓存联手:多级缓存架构的奥秘

多级缓存(如Redis+本地缓存)是一种在系统架构中广泛应用的提高系统性能和响应速度的技术手段,它综合利用了不同类型缓存的优势,以下为你详细介绍:基本概念本地缓存:指的是在应用程序所在的服务器内...

腾讯云国际站:腾讯云服务器如何配置Redis缓存?

本文由【云老大】TG@yunlaoda360撰写一、安装Redis使用包管理器安装(推荐)在CentOS系统中,可以通过yum包管理器安装Redis:sudoyumupdate-...

Spring Boot3 整合 Redis 实现数据缓存,你做对了吗?

你是否在开发互联网大厂后端项目时,遇到过系统响应速度慢的问题?当高并发请求涌入,数据库压力剧增,响应时间拉长,用户体验直线下降。相信不少后端开发同行都被这个问题困扰过。其实,通过在SpringBo...

【Redis】Redis应用问题-缓存穿透缓存击穿、缓存雪崩及解决方案

在我们使用redis时,也会存在一些问题,导致请求直接打到数据库上,导致数据库挂掉。下面我们来说说这些问题及解决方案。1、缓存穿透1.1场景一个请求进来后,先去redis进行查找,redis存在,则...

Spring boot 整合Redis缓存你了解多少

在前一篇里面讲到了Redis缓存击穿、缓存穿透、缓存雪崩这三者区别,接下来我们讲解Springboot整合Redis中的一些知识点:之前遇到过,有的了四五年,甚至更长时间的后端Java开发,并且...

揭秘!Redis 缓存与数据库一致性问题的终极解决方案

在现代软件开发中,Redis作为一款高性能的缓存数据库,被广泛应用于提升系统的响应速度和吞吐量。然而,缓存与数据库之间的数据一致性问题,一直是开发者们面临的一大挑战。本文将深入探讨Redis缓存...

高并发下Spring Cache缓存穿透?我用Caffeine+Redis破局

一、什么是缓存穿透?缓存穿透是指查询一个根本不存在的数据,导致请求直接穿透缓存层到达数据库,可能压垮数据库的现象。在高并发场景下,这尤其危险。典型场景:恶意攻击:故意查询不存在的ID(如负数或超大数值...

Redis缓存三剑客:穿透、雪崩、击穿—手把手教你解决

缓存穿透菜小弟:我先问问什么是缓存穿透?我听说是缓存查不到,直接去查数据库了。表哥:没错。缓存穿透是指查询一个缓存中不存在且数据库中也不存在的数据,导致每次请求都直接访问数据库的行为。这种行为会让缓存...

Redis中缓存穿透问题与解决方法

缓存穿透问题概述在Redis作为缓存使用时,缓存穿透是常见问题。正常查询流程是先从Redis缓存获取数据,若有则直接使用;若没有则去数据库查询,查到后存入缓存。但当请求的数据在缓存和数据库中都...

Redis客户端缓存的几种实现方式

前言:Redis作为当今最流行的内存数据库和缓存系统,被广泛应用于各类应用场景。然而,即使Redis本身性能卓越,在高并发场景下,应用于Redis服务器之间的网络通信仍可能成为性能瓶颈。所以客户端缓存...

Nginx合集-常用功能指导

1)启动、重启以及停止nginx进入sbin目录之后,输入以下命令#启动nginx./nginx#指定配置文件启动nginx./nginx-c/usr/local/nginx/conf/n...

腾讯云国际站:腾讯云怎么提升服务器速度?

本文由【云老大】TG@yunlaoda360撰写升级服务器规格选择更高性能的CPU、内存和带宽,以提供更好的处理能力和网络性能。优化网络配置调整网络接口卡(NIC)驱动,优化TCP/IP参数...

雷霆一击服务器管理员教程

本文转载莱卡云游戏服务器雷霆一击管理员教程(搜索莱卡云面版可搜到)首先你需要给服务器设置管理员密码,默认是空的管理员密码在启动页面进行设置设置完成后你需要重启服务器才可生效加入游戏后,点击键盘左上角E...

取消回复欢迎 发表评论: