这是一篇博文翻译,略有删减,整理代码方便统一阅读,Github链接:https://github.com/lesterli/rust-practice/tree/master/head-first/async-primer。
原文在2月11号的【Rust日报】中给大家推荐过, 原文链接: https://omarabid.com/async-rust
本文并不全面介绍Rust异步主题。如果对新的async/await关键字Futures感到疑惑,并且对Tokio的用途很感兴趣,那么到最后应该会不再毫无头绪。
Rust异步技术是Rust领域的新热点,它被誉为Rust的重要里程碑,特别适合开发高性能网络应用程序的人们。
让我们从头开始。
什么是异步?
关于Async,我给一个简短的版本:如果有一个处理器,想同时执行(类似)两项任务,将如何做?解决方案是先运行第一个任务,然后切换并运行第二个任务,然后再切换回去,依此类推,直到完成两个任务。
如果想给人以计算机同时运行两个任务的感觉(即多任务处理),则此功能很有用。另一个用例是IO操作。当程序等待网络响应时,CPU处于空闲状态。这是切换到另一个任务的理想时间。
那么我们如何编写异步代码?
首先,让我们从一些同步代码开始。
同步代码
让我们做一个简单的程序,该程序读取两个文件:file1.txt
和file2.txt
。我们从file1.txt
开始,然后移至file2.txt
。
我们将程序分为两个文件:main.rs
和file.rs
。file.rs
有一个函数:read_file
,在main.rs
中,用每个文件的路径为参数调用此函数。参见下面代码:
// sync-example/src/file.rs
use std::fs::File;
use std::io::{self, Read};
pub fn read_file(path: &str) -> io::Result<String> {
let mut file = File::open(path)?;
let mut buffer = String::new();
file.read_to_string(&mut buffer)?;
Ok(buffer)
}
// sync-example/src/main.rs
use std::io;
mod file;
fn main() -> io::Result<()> {
println!("program started");
let file1 = file::read_file("src/file1.txt")?;
println!("processed file 1");
let file2 = file::read_file("src/file2.txt")?;
println!("processed file 2");
dbg!(&file1);
dbg!(&file2);
Ok(())
}
使用cargo run
编译并运行程序。该程序应该毫无意外地运行,但是请确保已在src
文件夹中放置了两个文件(file1.txt
和file2.txt
)。
program started
processed file 1
processed file 2
[src/main.rs:14] &file1 = "file1"
[src/main.rs:15] &file2 = "file2"
到目前为止,一切都很好。如果需要在处理file2.txt
之前先处理file1.txt
,那么这是唯一的方法。但是有时不必关心每个文件的处理顺序。理想情