rust学习十二、测试

测试从来不是一件简单的事情,我本人深有体会!

书本作者引用了很重要的话:软件测试是证明 bug 存在的有效方法,而证明其不存在时则显得令人绝望的不足  (Edsger W. Dijkstra 在其 1972 年的文章【谦卑的程序员】(“The Humble Programmer”))

注:Edsger W. Dijkstra在1972获得图灵奖

 

本人学过一些语言,看过不少类似书籍,但是在书籍中间部分就让入门者注意测试,的确是挺特别的。

这充分地体现了作者对测试的重视。

既然专家都这么注意测试,我辈自当也需要格外注意。Rust由于其语言特性,应该是需要更加重视测试!

 

把测试当做无足轻重,或者认为不需要什么技术水平的,都是不专业的行为!it新人或者一些水平有问题的常常会有这个方面的看法!

本章节的内容比较多,但只要按照作者的例子作一遍,那么基本能够掌握!

rust测试需要用到cargo test命令,这个命令有许多选项,部分会在本章节中提到,还有许多是没有提到的。

把test命令的各个选项都熟悉下,也需要耗费不少时间!

 

本文所涉及到内容都是入门级别的,属于基本的粗浅的。但是掌握这些基本上已经可以做需要的测试了。

至于测试技巧,是另外一门学问了,和rust没有关系。

 

一、编写单元测试代码

本例模仿书上的,在库中编写单元测试代码。

按照惯例:创建一个tests模块,并在tests下编写测试代码

关键词:

#[cfg(test)] -- 告诉编译器运行的时候不要编辑

#[test]   --  告诉编译器这是一个测试单元

#[should_panic] - 告诉测试程序,如果不出意外就是失败

函数: assert_eq!,assert_ne!,assert!

二、控制测试

关键问题:

  1. 如何控制并发
  2. 如何控制需要测试的
  3. 如何忽略,或者如何只测试被标记为忽略的

控制并发

默认是多线程,但也可以通过一下方式指定需要的线程

-- --test-threads=n

控制测试目标

cargo test xxx ,xxx是测试的名称,即单元测试方法的名称.属于模糊匹配
cargo test --test xxx 指定集成测试的文件名,模糊匹配

忽略

#[ingore] -- 用于标记测试方法为被忽略

cargo test -- --ignored 只测试被忽略的

三、测试的组织结构

如何做单元测试

固定的规范-- 使用#[cfg(test)] 标记测试代码,并使用#[test]标记测试函数。其次模块通常也命名位tests

如何做集成测试

创建和src同i的目录tests,在tests下放任意个rs,每个rs都是一个单元包

每个单元包都可以应用tests的公共模块。

 

集成测试中的公共模块

使用老的模块定义规范:目录+mod.rs

如上图,haha模块在tests/haha下,其下有mod.rs

 

四、完整测试代码和示例

这个例子是基于一个库项目,有三个文件:

src/lib.rs 库 ,单元测试文件

tests/itest1.rs 集成测试文件之一

tests/haha/mod.rs 集成测试的模块haha

lib.rs

pub fn add(left: u64, right: u64) -> u64 {
    left + right
}

#[cfg(test)]
mod tests {
    use super::add;
    //测试宏 assert_eq!
    #[test]
    fn it_works1() {
        let result = add(2, 2);
        assert_eq!(result, 4);
    }

    //测试宏 assert!.并测试自定义的错误信息
    #[test]
    fn it_works2() {
        let result = add(1, 3);
        assert!(result==5,"1+3={},而不是5",result);
    }
    //测试宏 assert_ne!
    #[test]
    fn it_works3() {
        let result = add(1, 4);
        println!("这一句将在--show-output 的时候打印:1+4={}",result);
        assert_ne!(result,6);
    }
    //测试fail! 
    #[test]
    #[should_panic]
    fn it_fail() {
        //fail!("this test will fail");
        let result = add(1, 4);
        assert_eq!(result,6);
    }

    #[test]
    #[should_panic]
    fn access_outof_range() {
        let mut v = vec![1,2];  
        v[2] = 3;
    }    

    //如果返回Err,则测试失败
    #[test]
    fn test_result()->Result<(),String> {
        if 1==2 {
            Ok(())
        } else {
            Err("error".to_string())
        }
    }

    #[test]
    #[ignore]
    fn test_ignore() {
        assert_eq!(1,2, "1!={}",2);
    }
}

 

itest1.rs

use lzfmath::*;
mod haha;

#[test]
fn itest_add() {
    haha::haha();
    assert_eq!(add(1, 2), 3);
}

 

mod.rs

pub fn haha(){
    println!("哈哈!");
}

 

 

4.1标准单元测试结果

4.2设置并发和查看标准输出

 

4.3只测试it开头的单元测试

 

4.4只测试被忽略的

 

4.5集成测试+集成测试模块

 

五、小结

   代码测试主要有三个部分内容:
   1.如何编写测试
     主要涉及到几个编译标记/指令和几个函数/宏,Result
     编译指令
       a.#[cfg(test)]     一个条件编译指令,用于指定紧随其后的代码块或项(如函数、模块等)仅在测试环境下编译
       b.#[test]          函数测试标记,cargo的测试的时候,会找出有这些标记的函数,并执行测试
       c.#[should_panic]  函数测试标记,标记的函数会期待panic,如果函数没有panic就会测试失败。
     函数
       a.assert!      断定表达式为真。如果为真就是成功
       b.assert_eq!   断定两个值是否相等。如果相等就是成功
       c.assert_ne!   断定两个值是否不相等。如果不相等就是成功
       d.assert_matches! 断定表达式匹配模式。如果匹配就是成功.特别适用于枚举和结构等类型
       f.fail!        断定测试失败,并打印出字符串
       assert类函数,可以在标准参数之外再传递其它参数,比如失败时打印的字符串。rust会调用格式化字符串,并打印出参数。
     Result类型
       a.Result<T,E>  枚举类型,表示成功或失败。T为成功时的类型,E为错误类型
   2.如何控制测试
     默认的测试行为
       a.cargo test 在测试模式下编译代码并运行生成的测试二进制文件
       b.cargo test 产生的二进制文件的默认行为是并发运行所有的测试,并截获测试运行过程中产生的输出,阻止它们被显示出来,使得阅读测试结果相关的内容变得更容易
     
     如何控制(控制cargo test的行为)
       a.控制并发  -- --test-threads=1  只允许一个线程运行测试
       b.控制输出  -- --show-output   显示测试运行过程中产生的输出。这样可以输出通过部分的打印输出
       c.控制测试   <name> 只运行指定名称的测试 。注意这是模糊匹配   
                   add  测试包含add的测试。即cargo test add
                  --test xxx 指定具体文件名  cargo test --test itest1    测试文件名带itest1的测试
     忽略项目
       a.#[ignore]   在代码中标记需要忽略的内容
       b.测试被忽略项目  -- --ignored    ,这样只会测试被标记为#[ignore]的内容
 
     示例
       cargo test result  -- --test-threads=2  --show-output   在开启两个线程的情况下,测试包含result的测试,同时那些产生的输出也会显示出来
       cargo test -- --ignored  测试被标记为#[ignore]的内容
   3.测试的组织结构  
     如何作单元测试
       a.固定的规范-- 使用#[cfg(test)] 标记测试代码,并使用#[test]标记测试函数。其次模块通常也命名位tests
       
     如何作集成测试
       a.在项目下创建tests目录,然后在该目录下创建测试文件  。注意tests目录和src是同级的. 这个目录应该是可以是别的,也许可以通过test的选项来指定
       b.在测试目录tests可以创建任意多个文件,cargo会把每一个文件当做一个单元包进行测试
       c.运行 cargo test即可,但是这样会把单元测试一起执行。
       d.通过指定集成测试文件的方式,可以只运行指定的测试文件 cargo test --test itest1.也可以通过指定测试名称的方式
         carggo test xxx   
       综合起来,cargo test --test xxx 也许是更好的选择。
     
     集成测试的公共模块
       a.必须先创建一个目录,假设是haha
       b.在haha下创建mod.rs文件
       c.tests下的其它rs可以调用haha,并且haha不会被测试。haha的内容只会被集成测试代码调用。
       也就是说集成测试目录下的模块组织方式,反而使用的是旧的模块定义规范:基于目录+mod.rs
 
     为什么很多项目既有src/main.rs,src/lib.rs
       这都是为了方便测试。
       引入在集成测试中导入src/main.rs的模块并不容易。 main.rs是一个二进制文件,而lib.rs是库。

 

posted @ 2024-11-27 19:26  正在战斗中  阅读(14)  评论(0编辑  收藏  举报