步骤 1:编写失败的测试
我们遵循测试驱动开发(tdd)方法,首先编写一个测试来验证不区分大小写搜索功能。由于我们尚未实现该功能,因此这个测试会失败。
以下是测试代码:
#[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) ); } }
在上述代码中:
case_sensitive
测试验证了原始的区分大小写的搜索。case_insensitive
测试验证了我们将在后面实现的不区分大小写的搜索。此时,由于我们还没有实现search_case_insensitive
函数,编译将失败。
步骤 2:实现 search_case_insensitive 函数
为了实现不区分大小写的搜索,我们需要修改 search
函数,使其能够忽略大小写。
以下是 search_case_insensitive
函数的实现:
// 不区分大小写的搜索函数 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 // 返回匹配的行 }
代码解释:
query.to_lowercase()
将查询字符串转换为小写,以确保不区分大小写地进行匹配。line.to_lowercase()
将每一行内容转换为小写,并检查是否包含查询字符串。
步骤 3:修改 run 函数
run
函数是程序的核心,它负责执行搜索并输出结果。我们需要修改 run
函数,使其根据 config
结构体中的 ignore_case
字段来决定是使用区分大小写的 search
函数,还是不区分大小写的 search_case_insensitive
函数。
// 根据配置运行搜索 pub fn run(config: config) -> result<(), box<dyn error>> { let contents = fs::read_to_string(config.file_path)?; // 读取文件内容 // 根据 ignore_case 字段决定使用哪种搜索 let results = if config.ignore_case { search_case_insensitive(&config.query, &contents) // 不区分大小写的搜索 } else { search(&config.query, &contents) // 区分大小写的搜索 }; // 打印匹配的行 for line in results { println!("{line}"); } ok(()) }
代码解释:
config
结构体包含一个ignore_case
字段,用于控制是否启用不区分大小写的搜索。- 根据
ignore_case
的值,我们决定调用search
还是search_case_insensitive
。
步骤 4:获取环境变量
现在,我们需要从环境变量中获取 ignore_case
的值,来决定是否启用不区分大小写的搜索。
我们使用 rust 标准库中的 env
模块来访问环境变量。
use std::env; impl config { // 构建 config 实例 pub fn build(args: &[string]) -> result<config, &'static str> { if args.len() < 3 { return err("not enough arguments"); } let query = args[1].clone(); let file_path = args[2].clone(); // 检查环境变量 ignore_case 是否设置 let ignore_case = env::var("ignore_case").is_ok(); // 如果环境变量存在,则启用不区分大小写的搜索 ok(config { query, file_path, ignore_case, }) } }
代码解释:
- 我们使用
env::var("ignore_case").is_ok()
来检查环境变量ignore_case
是否被设置。 - 如果设置了该环境变量,我们将
ignore_case
设置为true
,否则默认为false
。
步骤 5:运行程序
现在,我们的程序已经支持根据环境变量来控制搜索模式。你可以通过以下命令来运行程序:
- 不使用环境变量,执行区分大小写的搜索:
$ cargo run -- to poem.txt
- 设置环境变量
ignore_case=1
,启用不区分大小写的搜索:
$ ignore_case=1 cargo run -- to poem.txt
对于 powershell 用户,可以使用以下命令:
ps> $env:ignore_case=1; cargo run -- to poem.txt
总结
通过使用环境变量,我们成功地在rust程序中实现了不区分大小写的搜索功能。
这种方式让用户可以通过简单的环境变量配置来改变程序的行为,而不需要每次运行程序时都指定命令行参数。
通过这种方法,我们可以在命令行工具中灵活地控制不同的搜索模式,使程序更加友好和易于配置。
在本教程中,我们:
- 通过 tdd 编写了失败的测试,并逐步实现了功能。
- 学会了如何使用 rust 标准库中的
env
模块获取环境变量。 - 通过环境变量控制程序行为,让命令行工具更加灵活。
通过这个示例,希望你对 rust 中如何使用环境变量有所了解,并能够在自己的项目中灵活应用。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持代码网。
发表评论