17. 一个I/O项目:构建命令行程序(下)

一、采用测试驱动开发完善库的功能

1.1 编写失败测试用例

  1. 在lib.rs中写一个简单的search函数
pub fn search<'a>(query: &str, contents: &'a str) -> Vec<&'a str>{
    vec![]
}
  • 目前该函数是一个总是返回空的vector函数;
  • 函数的参数是&str型,因此需要显式地定义生命周期;
  • 生命周期指定contents的生命周期与返回值的相关连;
  1. 在lib.rs中编写测试代码
#[cfg(test)]
mod tests{
    use super::*;

    #[test]
    fn one_result(){
        let query = "duct";
        let contents = "\
Rust:
safe, fast, productive.
Pick there.";
        
        assert_eq!(vec!["safe, fast, productive."],
        search(query, contents)
        )
    }
}
  1. 运行测试会如预期的失败了;

1.2 编写成功测试用例

失败的原因在于search返回空,因此完善它就行

pub fn search<'a>(query: &str, contents: &'a str) -> Vec<&'a str>{
    let mut results = Vec::new();

    for line in contents.lines(){
        if line.contains(query){
            results.push(line);
        }
    }

    results
}
  • 创建一个可变的Vector;
  • 使用字符串类的迭代方法一次读取一行;
  • 在此行中查找,找到后 push进入Vector即可;

编译测试用例
在这里插入图片描述

1.3 在run函数中打印搜索到的行

  • 修改run函数,将搜索到的行内容打印出来
pub fn run(config: Config) -> Result<(), Box<dyn Error>> {
    let contents = fs::read_to_string(config.filename)?;

    for line in search(&config.query, &contents) {
        println!("{}", line);
    }

    Ok(())
}

运行命令cargo run body poem.txt,旨在poem.txt文件内查找body字符串
在这里插入图片描述

二、添加大小写不敏感功能

  • 编写大小写不敏感的测试函数search_case_insensitive
  • 将旧的one_result改名为大小写敏感函数case_sensitive

大小写不敏感的search函数

pub fn search_case_insensitive<'a>(query: &str, contents: &'a str) -> Vec<&'a str> {
    let query = query.to_lowercase();
    let mut results = Vec::new();

    for line in contents.lines() {
        if line.to_lowercase().contains(&query) {
            results.push(line);
        }
    }

    results
}
  • 大小写不敏感的做法就是将读取到的字母以及目标子串全部改为大写或小写;

编写测试函数

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn case_sensitive() {
        let query = "duct";
        let contents = "\
Rust:
safe, fast, productive.
Pick three.
Duct tape.";

        assert_eq!(
            vec!["safe, fast, productive."],
            search(query, contents)
        );
    }

    #[test]
    fn case_insensitive() {
        let query = "rUsT";
        let contents = "\
Rust:
safe, fast, productive.
Pick three.
Trust me.";

        assert_eq!(
            vec!["Rust:", "Trust me."],
            search_case_insensitive(query, contents)
        );
    }
}

在Config中增加case_sensitive并修改构造函数

use std::env;

pub struct Config{
    query: String,
    filename: String,
    case_sensitive: bool, 
}

impl Config{
    pub fn new(args: &[String]) -> Result<Config, &'static str>{
        if args.len() < 3{
            return Err("Not enough arguments!");
        }

        let query = args[1].clone();
        let filename = args[2].clone();
        let case_sensitive = env::var("CASE_INSENSITIVE").is_err();

        Ok(Config {query, filename, case_sensitive})
    }
}
  • 在Config中添加了case_sensitive的布尔值,用来说明是否大小写敏感;
  • 在Config的new函数中使用env::var读取CASE_INSENSITIVE的环境变量设置情况;
  • 如果设置了CASE_INSENSITIVE的值is_err返回True就是大小写不敏感的,否则是敏感的;
  • 注意:不论CASE_INSENSITIVE设置为多少,都是大小写不敏感的

修改run函数,以判断是否起用大小写敏感

pub fn run(config: Config) -> Result<(), Box<dyn Error>> {
    let contents = fs::read_to_string(config.filename)?;

    let results = if config.case_sensitive {
        search(&config.query, &contents)
    }else{
        search_case_insensitive(&config.query, &contents)
    };

    for line in results{
        println!("{}", line);
    }

    Ok(())
}

测试,先不加环境变量测试大小写敏感的,再测试不敏感的。
在这里插入图片描述

三、将错误信息输出到标准错误

  • 通过println!打印的信息都输出到了终端;
  • 大部分终端都提供了两种输出:标准输出(standard output,stdout)和标准错误(standard error,stderr);
  • 标准输出对应一般信息,标准错误用于错误信息;
  • Rust提供eprintln!宏输出标准错误信息;

修改main的标准错误输出

fn main() {
    let args: Vec<String> = env::args().collect();
    
    let config = Config::new(&args).unwrap_or_else(|err| {
        eprintln!("Problem parsing arguments: {}", err);
        process::exit(1);
    });

    if let Err(e) = minigrep::run(config) {
        eprintln!("Application error: {}", e);

        process::exit(1);
    }
}

四、附录完整代码

main.rs

use std::env;
use std::process;
use minigrep::Config;


fn main() {
    let args: Vec<String> = env::args().collect();
    
    let config = Config::new(&args).unwrap_or_else(|err| {
        eprintln!("Problem parsing arguments: {}", err);
        process::exit(1);
    });

    if let Err(e) = minigrep::run(config) {
        eprintln!("Application error: {}", e);

        process::exit(1);
    }
}

lib.rs

use std::fs;
use std::env;
use std::error::Error;

pub struct Config{
    query: String,
    filename: String,
    case_sensitive: bool, 
}

impl Config{
    pub fn new(args: &[String]) -> Result<Config, &'static str>{
        if args.len() < 3{
            return Err("Not enough arguments!");
        }

        let query = args[1].clone();
        let filename = args[2].clone();
        let case_sensitive = env::var("CASE_INSENSITIVE").is_err();

        Ok(Config {query, filename, case_sensitive})
    }
}

pub fn run(config: Config) -> Result<(), Box<dyn Error>> {
    let contents = fs::read_to_string(config.filename)?;

    let results = if config.case_sensitive {
        search(&config.query, &contents)
    }else{
        search_case_insensitive(&config.query, &contents)
    };

    for line in results{
        println!("{}", line);
    }

    Ok(())
}

pub fn search<'a>(query: &str, contents: &'a str) -> Vec<&'a str>{
    let mut results = Vec::new();

    for line in contents.lines(){
        if line.contains(query){
            results.push(line);
        }
    }

    results
}

pub fn search_case_insensitive<'a>(query: &str, contents: &'a str) -> Vec<&'a str> {
    let query = query.to_lowercase();
    let mut results = Vec::new();

    for line in contents.lines() {
        if line.to_lowercase().contains(&query) {
            results.push(line);
        }
    }

    results
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn case_sensitive() {
        let query = "duct";
        let contents = "\
Rust:
safe, fast, productive.
Pick three.
Duct tape.";

        assert_eq!(
            vec!["safe, fast, productive."],
            search(query, contents)
        );
    }

    #[test]
    fn case_insensitive() {
        let query = "rUsT";
        let contents = "\
Rust:
safe, fast, productive.
Pick three.
Trust me.";

        assert_eq!(
            vec!["Rust:", "Trust me."],
            search_case_insensitive(query, contents)
        );
    }
}
  • 16
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

贱贱的剑

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值