包阅导读总结
1.
关键词:Rust、异步编程、Futures、async/await、tokio
2.
总结:本文介绍了 Rust 中的异步编程,包括何时使用、基础概念(Futures)、async/await 用法,以及在 main 函数中处理异步操作的两种方式(tokio 和 futures 库),指出异步编程在特定场景有优势,但只是基础概述。
3.
主要内容:
– 异步编程基础
– 与多线程对比,异步编程在某些场景优势明显,如网络请求和 IO 操作。
– 空闲时事件处理器可处理其他任务,而非阻塞线程。
– 异步 Rust 概述 – Futures
– Futures 类似 JavaScript 中的 promises,有两种 poll 状态。
– Futures 是惰性的,通过.await 运行。
– async/await 在 Rust 中的使用
– async 用于声明异步函数,.await 暂停执行直至 Future 结果准备好。
– 在 main 中处理异步操作
– Rust 不允许 async fn main 函数,可用 tokio 或 futures 库解决。
– tokio 提供工具和 API,可声明异步 main 函数。
– futures 库的 block_on 方法阻塞主线程直到 Future 结果准备好。
– 结论
– 异步编程更高效,但本文只是概述,还有其他相关构造。
思维导图:
文章地址:https://www.freecodecamp.org/news/how-asynchronous-programming-works-in-rust/
文章来源:freecodecamp.org
作者:Oduah Chigozie
发布时间:2024/8/15 18:24
语言:英文
总字数:1002字
预计阅读时间:5分钟
评分:83分
标签:异步编程,Rust,Futures,async/await,tokio
以下为原文内容
本内容来源于用户推荐转载,旨在分享知识与观点,如有侵权请联系删除 联系邮箱 media@ilingban.com
If you’re familiar with languages like JavaScript and Python, you may have heard about asynchronous programming. And perhaps you’re wondering how it works in Rust.
In this article, I’ll give you a simple overview of how asynchronous programming works in Rust. You’ll learn about futures as well as async
/.await
.
This article isn’t a beginner’s guide to Rust or asynchronous programming, so you’ll need to have some experience with programming in Rust and familiarity with asynchronous programming to get the most out of this guide.
With that said, let’s begin!
When Should You Use Asynchronous Programming?
Asynchronous tasks work like a more integrated version of threads. You can use them in a lot of the same scenarios where you can use multiple threads. The benefits async programming provides over multiple threads is that multi-threaded applications have a larger overhead to work on multiple tasks compared to asynchronous applications.
But this doesn’t make asynchronous applications better than multithreaded applications. Multi-threaded programs can run multiple computing-intensive tasks simultaneously – and multiple times faster than if you ran all the tasks in a single thread.
With asynchronous coding, trying to run multiple computing-intensive applications simultaneously will be much slower than just running every task on a single thread.
Asynchronous programming is very good for building applications that make many network requests or many IO operations like file reading and writing.
I can’t cover every case when you’ll want to use asynchronous techniques. But I can say that it’s most beneficial for tasks that have a lot of idle time (for example, waiting for a server to respond to a network request).
During the idle time of an asynchronous task, instead of blocking the thread, the event handler works on other tasks in the program until the asynchronous task is ready to make progress.
Overview of Asynchronous Rust – Futures
The basics of asynchronous Rust are Futures. Futures are similar to promises in JavaScript. They are Rust’s way of saying “hey, I’m going to give you the result later, but just hold on to this (the future instance) to keep track of if the result is ready.”
Futures are traits, with a poll
state that is either Poll::Pending
to signify that the future is in the process of executing its task, or Poll::Ready
to signify that the future is done with its task.
Futures are lazy. They run when you .await
them (we’ll look into how to .await
them in the next section). .await
ing a future pauses the execution of an asynchronous thread, and begins running its task. At this point the result of the poll
method is Poll::Pending
. When the future is done with its task, poll
‘s state will become Poll::Ready
, and the future will allow it’s thread to proceed.
If you want to get more into the technical details about Futures, you can check out the documentation.
async/.await in Rust
async
and .await
are the primary ways you’ll be working with asynchronous code in Rust. async
is a keyword for declaring asynchronous functions. Within those functions, the .await
keyword pauses its execution until the result of the future is ready.
Let’s take a look at an example:
async fn add(a: u8, b: u8) -> u8 { a + b}async fn secondFunc() -> u8 { let a = 10; let b = 20; let result = add(a, b).await; return result;}
Any asynchronous function declared with async fn
wraps its return value in a future. On the third line of secondFunc
, we .await
the future from add(a, b)
to get its result before proceeding to return it.
How to Work with Asynchronous Operations in main
Rust doesn’t allow you to async fn
main functions. Running asynchronous operations from a non-asynchronous function may lead to some operations not fully concluding before the main thread is exited.
In this section, we’ll look at two ways solve this issue. The two ways to do this are with futures
and tokio
. Let’s look at both.
tokio
tokio
is a platform that provides tools and APIs for performing asynchronous applications. tokio
also allows you to easily declare an asynchronous main function, which will help with the issue I described earlier.
To install tokio
in your project, add this line to its Cargo.toml
:
[dependencies]tokio = { version = "1", features = ["full"] }
After adding the line, you can write your main
functions like this:
async fn add(a: u8, b: u8) -> u8 { a + b}#[tokio::main]async fn main() { let a = 10; let b = 20; let result = add(a, b).await; println!("{result}");}
The futures
library
futures
is a library that provides methods for working with asynchronous Rust. For our use case, futures
provides futures::executor::block_on()
. This method works similar to .await
in asynchronous functions. It blocks the main thread, until the result of future is ready. futures::executor::block_on()
also returns the result of the completed future.
To install futures
in your project, add this line to its Cargo.toml
:
[dependencies]futures = "0.3"
After installing the library, you can do something like this:
use futures::executor::block_on;async fn add(a: u8, b: u8) -> u8 { a + b}fn main() { let a = 10; let b = 20; let result = block_on(add(a, b)); println!("{result}"); }
First, we import the block_on
method on the first line and use the method to block the main thread until the result of the add(a, b)
function was ready. We also didn’t have to make the main
function asynchronous like we did with tokio
.
Conclusion
Asynchronous programming allows you to run operations in parallel and with less overhead and complexity compared to traditional multi-threading. In Rust, it allows you to build I/O and network applications more efficiently.
While this article should help you become familiar with the basics asynchronous programming in Rust, it’s still just an overview. There are some cases where you’ll need to use other constructs in Rust like Pinning, Arcs, and so on.
If you have any questions or thoughts, feel free to reach out to me. Thanks for reading!