打开 I∕O 项目的 src∕main.rs 文件,它看起来应该像这样:# use std::env;

use std::process;

use minigrep::Config;

fn main() { let args: Vec = env::args().collect(); let config = Config::new(&args).unwrap_or_else(|err| { eprintln!(“Problem parsing arguments: {}”, err); process::exit(1); }); // --snip–

if let Err(e) = minigrep::run(config) {

eprintln!(“Application error: {}”, e);

process::exit(1);

}

} 修改第十二章结尾示例 12-24 中的 main 函数的开头为示例 13-25 中的代码。在更新 Config::new 之前 这些代码还不能编译:

use std::env;

use std::process;

use minigrep::Config;

fn main() { let config = Config::new(env::args()).unwrap_or_else(|err| { eprintln!(“Problem parsing arguments: {}”, err); process::exit(1); }); // --snip–

if let Err(e) = minigrep::run(config) {

eprintln!(“Application error: {}”, e);

process::exit(1);

}

} 将 env::args 的返回值传递给 Config::new env::args 函数返回一个迭代器!不同于将迭代器的值收集到一个 vector 中接着传递一个 slice 给 Config::new,现在我们直接将 env::args 返回的迭代器的所有权传递给 Config::new。 接下来需要更新 Config::new 的定义。在 I∕O 项目的 src∕lib.rs 中,将 Config::new 的签名改为 这仍然不能编译因为我们还需更新函数体:

use std::env;

use std::error::Error;

use std::fs;

pub struct Config {

pub query: String,

pub filename: String,

pub case_sensitive: bool,

}

impl Config { pub fn new(mut args: env::Args) -> Result { // --snip–

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> {

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)

);

}

}

以迭代器作为参数更新 Config::new 的签名 env::args 函数的标准库文档显示,它返回的迭代器的类型为 std:: env::Args。我们已经更新了 Config :: new 函数的签名,因此参数 args 的类型为 std:: env::Args 而不是 &[String]。因为我们拥有 args 的所有权,并 且将通过对其进行迭代来改变 args ,所以我们可以将 mut 关键字添加到 args 参数的规范中以使其可变。 现在我们还需指定字符串 slice 错误类型只能有 ’ static 生命周期。因为我们之前只会返回字符串 slice, 所以这是成立的。然而,当参数中有一个引用的时候,返回类型的引用有可能与参数的引用有着相同的 生命周期。之前第十章 ” 生命周期省略” 部分讨论的规则生效,因此无需注明 &str 的生命周期。随着对 args 的修改,生命周期省略规则不再适用,所以必须指定 ’ static 生命周期。

参考阅读

评论可见,请评论后查看内容,谢谢!!!
 您阅读本篇文章共花了: