1//! Asynchronous green-threads.
2//!
3//! ## What are Tasks?
4//!
5//! A _task_ is a light weight, non-blocking unit of execution. A task is similar
6//! to an OS thread, but rather than being managed by the OS scheduler, they are
7//! managed by the [Tokio runtime][rt]. Another name for this general pattern is
8//! [green threads]. If you are familiar with [`Go's goroutines`], [`Kotlin's
9//! coroutines`], or [`Erlang's processes`], you can think of Tokio's tasks as
10//! something similar.
11//!
12//! Key points about tasks include:
13//!
14//! * Tasks are **light weight**. Because tasks are scheduled by the Tokio
15//! runtime rather than the operating system, creating new tasks or switching
16//! between tasks does not require a context switch and has fairly low
17//! overhead. Creating, running, and destroying large numbers of tasks is
18//! quite cheap, especially compared to OS threads.
19//!
20//! * Tasks are scheduled **cooperatively**. Most operating systems implement
21//! _preemptive multitasking_. This is a scheduling technique where the
22//! operating system allows each thread to run for a period of time, and then
23//! _preempts_ it, temporarily pausing that thread and switching to another.
24//! Tasks, on the other hand, implement _cooperative multitasking_. In
25//! cooperative multitasking, a task is allowed to run until it _yields_,
26//! indicating to the Tokio runtime's scheduler that it cannot currently
27//! continue executing. When a task yields, the Tokio runtime switches to
28//! executing the next task.
29//!
30//! * Tasks are **non-blocking**. Typically, when an OS thread performs I/O or
31//! must synchronize with another thread, it _blocks_, allowing the OS to
32//! schedule another thread. When a task cannot continue executing, it must
33//! yield instead, allowing the Tokio runtime to schedule another task. Tasks
34//! should generally not perform system calls or other operations that could
35//! block a thread, as this would prevent other tasks running on the same
36//! thread from executing as well. Instead, this module provides APIs for
37//! running blocking operations in an asynchronous context.
38//!
39//! [rt]: crate::runtime
40//! [green threads]: https://en.wikipedia.org/wiki/Green_threads
41//! [Go's goroutines]: https://tour.golang.org/concurrency/1
42//! [Kotlin's coroutines]: https://kotlinlang.org/docs/reference/coroutines-overview.html
43//! [Erlang's processes]: http://erlang.org/doc/getting_started/conc_prog.html#processes
44//!
45//! ## Working with Tasks
46//!
47//! This module provides the following APIs for working with tasks:
48//!
49//! ### Spawning
50//!
51//! Perhaps the most important function in this module is [`task::spawn`]. This
52//! function can be thought of as an async equivalent to the standard library's
53//! [`thread::spawn`][`std::thread::spawn`]. It takes an `async` block or other
54//! [future], and creates a new task to run that work concurrently:
55//!
56//! ```
57//! use tokio::task;
58//!
59//! # async fn doc() {
60//! task::spawn(async {
61//! // perform some work here...
62//! });
63//! # }
64//! ```
65//!
66//! Like [`std::thread::spawn`], `task::spawn` returns a [`JoinHandle`] struct.
67//! A `JoinHandle` is itself a future which may be used to await the output of
68//! the spawned task. For example:
69//!
70//! ```
71//! use tokio::task;
72//!
73//! # #[tokio::main] async fn main() -> Result<(), Box<dyn std::error::Error>> {
74//! let join = task::spawn(async {
75//! // ...
76//! "hello world!"
77//! });
78//!
79//! // ...
80//!
81//! // Await the result of the spawned task.
82//! let result = join.await?;
83//! assert_eq!(result, "hello world!");
84//! # Ok(())
85//! # }
86//! ```
87//!
88//! Again, like `std::thread`'s [`JoinHandle` type][thread_join], if the spawned
89//! task panics, awaiting its `JoinHandle` will return a [`JoinError`]. For
90//! example:
91//!
92//! ```
93//! use tokio::task;
94//!
95//! # #[tokio::main] async fn main() {
96//! let join = task::spawn(async {
97//! panic!("something bad happened!")
98//! });
99//!
100//! // The returned result indicates that the task failed.
101//! assert!(join.await.is_err());
102//! # }
103//! ```
104//!
105//! `spawn`, `JoinHandle`, and `JoinError` are present when the "rt"
106//! feature flag is enabled.
107//!
108//! [`task::spawn`]: crate::task::spawn()
109//! [future]: std::future::Future
110//! [`std::thread::spawn`]: std::thread::spawn
111//! [`JoinHandle`]: crate::task::JoinHandle
112//! [thread_join]: std::thread::JoinHandle
113//! [`JoinError`]: crate::task::JoinError
114//!
115//! #### Cancellation
116//!
117//! Spawned tasks may be cancelled using the [`JoinHandle::abort`] or
118//! [`AbortHandle::abort`] methods. When one of these methods are called, the
119//! task is signalled to shut down next time it yields at an `.await` point. If
120//! the task is already idle, then it will be shut down as soon as possible
121//! without running again before being shut down. Additionally, shutting down a
122//! Tokio runtime (e.g. by returning from `#[tokio::main]`) immediately cancels
123//! all tasks on it.
124//!
125//! When tasks are shut down, it will stop running at whichever `.await` it has
126//! yielded at. All local variables are destroyed by running their destructor.
127//! Once shutdown has completed, awaiting the [`JoinHandle`] will fail with a
128//! [cancelled error](crate::task::JoinError::is_cancelled).
129//!
130//! Note that aborting a task does not guarantee that it fails with a cancelled
131//! error, since it may complete normally first. For example, if the task does
132//! not yield to the runtime at any point between the call to `abort` and the
133//! end of the task, then the [`JoinHandle`] will instead report that the task
134//! exited normally.
135//!
136//! Be aware that calls to [`JoinHandle::abort`] just schedule the task for
137//! cancellation, and will return before the cancellation has completed. To wait
138//! for cancellation to complete, wait for the task to finish by awaiting the
139//! [`JoinHandle`]. Similarly, the [`JoinHandle::is_finished`] method does not
140//! return `true` until the cancellation has finished.
141//!
142//! Calling [`JoinHandle::abort`] multiple times has the same effect as calling
143//! it once.
144//!
145//! Tokio also provides an [`AbortHandle`], which is like the [`JoinHandle`],
146//! except that it does not provide a mechanism to wait for the task to finish.
147//! Each task can only have one [`JoinHandle`], but it can have more than one
148//! [`AbortHandle`].
149//!
150//! [`JoinHandle::abort`]: crate::task::JoinHandle::abort
151//! [`AbortHandle::abort`]: crate::task::AbortHandle::abort
152//! [`AbortHandle`]: crate::task::AbortHandle
153//! [`JoinHandle::is_finished`]: crate::task::JoinHandle::is_finished
154//!
155//! ### Blocking and Yielding
156//!
157//! As we discussed above, code running in asynchronous tasks should not perform
158//! operations that can block. A blocking operation performed in a task running
159//! on a thread that is also running other tasks would block the entire thread,
160//! preventing other tasks from running.
161//!
162//! Instead, Tokio provides two APIs for running blocking operations in an
163//! asynchronous context: [`task::spawn_blocking`] and [`task::block_in_place`].
164//!
165//! Be aware that if you call a non-async method from async code, that non-async
166//! method is still inside the asynchronous context, so you should also avoid
167//! blocking operations there. This includes destructors of objects destroyed in
168//! async code.
169//!
170//! #### `spawn_blocking`
171//!
172//! The `task::spawn_blocking` function is similar to the `task::spawn` function
173//! discussed in the previous section, but rather than spawning an
174//! _non-blocking_ future on the Tokio runtime, it instead spawns a
175//! _blocking_ function on a dedicated thread pool for blocking tasks. For
176//! example:
177//!
178//! ```
179//! use tokio::task;
180//!
181//! # async fn docs() {
182//! task::spawn_blocking(|| {
183//! // do some compute-heavy work or call synchronous code
184//! });
185//! # }
186//! ```
187//!
188//! Just like `task::spawn`, `task::spawn_blocking` returns a `JoinHandle`
189//! which we can use to await the result of the blocking operation:
190//!
191//! ```rust
192//! # use tokio::task;
193//! # async fn docs() -> Result<(), Box<dyn std::error::Error>>{
194//! let join = task::spawn_blocking(|| {
195//! // do some compute-heavy work or call synchronous code
196//! "blocking completed"
197//! });
198//!
199//! let result = join.await?;
200//! assert_eq!(result, "blocking completed");
201//! # Ok(())
202//! # }
203//! ```
204//!
205//! #### `block_in_place`
206//!
207//! When using the [multi-threaded runtime][rt-multi-thread], the [`task::block_in_place`]
208//! function is also available. Like `task::spawn_blocking`, this function
209//! allows running a blocking operation from an asynchronous context. Unlike
210//! `spawn_blocking`, however, `block_in_place` works by transitioning the
211//! _current_ worker thread to a blocking thread, moving other tasks running on
212//! that thread to another worker thread. This can improve performance by avoiding
213//! context switches.
214//!
215//! For example:
216//!
217//! ```
218//! use tokio::task;
219//!
220//! # async fn docs() {
221//! let result = task::block_in_place(|| {
222//! // do some compute-heavy work or call synchronous code
223//! "blocking completed"
224//! });
225//!
226//! assert_eq!(result, "blocking completed");
227//! # }
228//! ```
229//!
230//! #### `yield_now`
231//!
232//! In addition, this module provides a [`task::yield_now`] async function
233//! that is analogous to the standard library's [`thread::yield_now`]. Calling
234//! and `await`ing this function will cause the current task to yield to the
235//! Tokio runtime's scheduler, allowing other tasks to be
236//! scheduled. Eventually, the yielding task will be polled again, allowing it
237//! to execute. For example:
238//!
239//! ```rust
240//! use tokio::task;
241//!
242//! # #[tokio::main] async fn main() {
243//! async {
244//! task::spawn(async {
245//! // ...
246//! println!("spawned task done!")
247//! });
248//!
249//! // Yield, allowing the newly-spawned task to execute first.
250//! task::yield_now().await;
251//! println!("main task done!");
252//! }
253//! # .await;
254//! # }
255//! ```
256//!
257//! ### Cooperative scheduling
258//!
259//! A single call to [`poll`] on a top-level task may potentially do a lot of
260//! work before it returns `Poll::Pending`. If a task runs for a long period of
261//! time without yielding back to the executor, it can starve other tasks
262//! waiting on that executor to execute them, or drive underlying resources.
263//! Since Rust does not have a runtime, it is difficult to forcibly preempt a
264//! long-running task. Instead, this module provides an opt-in mechanism for
265//! futures to collaborate with the executor to avoid starvation.
266//!
267//! Consider a future like this one:
268//!
269//! ```
270//! # use tokio_stream::{Stream, StreamExt};
271//! async fn drop_all<I: Stream + Unpin>(mut input: I) {
272//! while let Some(_) = input.next().await {}
273//! }
274//! ```
275//!
276//! It may look harmless, but consider what happens under heavy load if the
277//! input stream is _always_ ready. If we spawn `drop_all`, the task will never
278//! yield, and will starve other tasks and resources on the same executor.
279//!
280//! To account for this, Tokio has explicit yield points in a number of library
281//! functions, which force tasks to return to the executor periodically.
282//!
283//!
284//! #### unconstrained
285//!
286//! If necessary, [`task::unconstrained`] lets you opt a future out of Tokio's cooperative
287//! scheduling. When a future is wrapped with `unconstrained`, it will never be forced to yield to
288//! Tokio. For example:
289//!
290//! ```
291//! # #[tokio::main]
292//! # async fn main() {
293//! use tokio::{task, sync::mpsc};
294//!
295//! let fut = async {
296//! let (tx, mut rx) = mpsc::unbounded_channel();
297//!
298//! for i in 0..1000 {
299//! let _ = tx.send(());
300//! // This will always be ready. If coop was in effect, this code would be forced to yield
301//! // periodically. However, if left unconstrained, then this code will never yield.
302//! rx.recv().await;
303//! }
304//! };
305//!
306//! task::unconstrained(fut).await;
307//! # }
308//! ```
309//!
310//! [`task::spawn_blocking`]: crate::task::spawn_blocking
311//! [`task::block_in_place`]: crate::task::block_in_place
312//! [rt-multi-thread]: ../runtime/index.html#threaded-scheduler
313//! [`task::yield_now`]: crate::task::yield_now()
314//! [`thread::yield_now`]: std::thread::yield_now
315//! [`task::unconstrained`]: crate::task::unconstrained()
316//! [`poll`]: method@std::future::Future::poll
317
318cfg_rt! {
319 pub use crate::runtime::task::{JoinError, JoinHandle};
320
321 cfg_not_wasi! {
322 mod blocking;
323 pub use blocking::spawn_blocking;
324 }
325
326 mod spawn;
327 pub use spawn::spawn;
328
329 cfg_rt_multi_thread! {
330 pub use blocking::block_in_place;
331 }
332
333 mod yield_now;
334 pub use yield_now::yield_now;
335
336 cfg_unstable! {
337 mod consume_budget;
338 pub use consume_budget::consume_budget;
339 }
340
341 mod local;
342 pub use local::{spawn_local, LocalSet, LocalEnterGuard};
343
344 mod task_local;
345 pub use task_local::LocalKey;
346
347 mod unconstrained;
348 pub use unconstrained::{unconstrained, Unconstrained};
349
350 #[doc(inline)]
351 pub use join_set::JoinSet;
352 pub use crate::runtime::task::AbortHandle;
353
354 // Uses #[cfg(...)] instead of macro since the macro adds docsrs annotations.
355 #[cfg(not(tokio_unstable))]
356 mod join_set;
357 #[cfg(tokio_unstable)]
358 pub mod join_set;
359
360 cfg_unstable! {
361 pub use crate::runtime::task::{Id, id, try_id};
362 }
363
364 cfg_trace! {
365 mod builder;
366 pub use builder::Builder;
367 }
368
369 /// Task-related futures.
370 pub mod futures {
371 pub use super::task_local::TaskLocalFuture;
372 }
373}
374

Provided by KDAB

Privacy Policy