Pin
Future์ ๋ํด await
๋ฅผ ํธ์ถํ์ฌ ๊ทธ future๊ฐ ์ค๋น๋๊ธฐ๋ฅผ ๊ธฐ๋ค๋ฆด ๋, ๋ชจ๋ ๋ก์ปฌ ๋ณ์(์ผ๋ฐ์ ์ผ๋ก ์คํ ํ๋ ์์ ์ ์ฅ๋จ)๋ ๊ทธ future๊ฐ์ฒด ์์ ์ ์ฅ๋ฉ๋๋ค. ๋ง์ฝ ๊ทธ future์ ์คํ์ ์๋ ์ด๋ค ๋ฐ์ดํฐ๋ก์ ํฌ์ธํฐ๊ฐ ์์ผ๋ฉด ์ด๋ฌํ ํฌ์ธํฐ๋ ์ฌ๋ฐ๋ฅด์ง ์์ ์ ์์ต๋๋ค. ์์ ํ์ง ์์ต๋๋ค.
๋ฐ๋ผ์ future๊ฐ ๊ฐ๋ฆฌํค๋ ์ฃผ์๊ฐ ๋ณ๊ฒฝ๋์ง ์๋๋ก ํด์ผ ํฉ๋๋ค. ์ด๊ฒ์ด future๋ฅผ pin
(๊ณ ์ )ํด์ผ ํ๋ ์ด์ ์
๋๋ค. select!
์์ ๋์ผํ future๋ฅผ ๋ฐ๋ณต์ ์ผ๋ก ์ฌ์ฉํ๋ฉด ๊ณ ์ ๊ฐ ๋ฌธ์ ๊ฐ ๋ฐ์ํ๋ ๊ฒฝ์ฐ๊ฐ ๋ง์ต๋๋ค.
use tokio::sync::{mpsc, oneshot}; use tokio::task::spawn; use tokio::time::{sleep, Duration}; // A work item. In this case, just sleep for the given time and respond // with a message on the `respond_on` channel. #[derive(Debug)] struct Work { input: u32, respond_on: oneshot::Sender<u32>, } // A worker which listens for work on a queue and performs it. async fn worker(mut work_queue: mpsc::Receiver<Work>) { let mut iterations = 0; loop { tokio::select! { Some(work) = work_queue.recv() => { sleep(Duration::from_millis(10)).await; // Pretend to work. work.respond_on .send(work.input * 1000) .expect("failed to send response"); iterations += 1; } // TODO: report number of iterations every 100ms } } } // A requester which requests work and waits for it to complete. async fn do_work(work_queue: &mpsc::Sender<Work>, input: u32) -> u32 { let (tx, rx) = oneshot::channel(); work_queue .send(Work { input, respond_on: tx, }) .await .expect("failed to send on work queue"); rx.await.expect("failed waiting for response") } #[tokio::main] async fn main() { let (tx, rx) = mpsc::channel(10); spawn(worker(rx)); for i in 0..100 { let resp = do_work(&tx, i).await; println!("work result for iteration {i}: {resp}"); } }
-
์์์ ์๊ฐํ ๊ฒ์ ์กํฐ(actor) ํจํด์ ํ ์๋ผ๊ณ ๋ด๋ ๋ฌด๋ฐฉํฉ๋๋ค. ์กํฐ๋ ์ผ๋ฐ์ ์ผ๋ก ๋ฃจํ ์์์
select!
๋ฅผ ํธ์ถํฉ๋๋ค. -
์ด์ ๊ฐ์ ๋ช ๊ฐ์ ๋ด์ฉ์ ์์ฝํ ๊ฒ์ด๊ธฐ ๋๋ฌธ์ ์ฒ์ฒํ ์ดํด๋ณด์ธ์.
-
_ = sleep(Duration::from_millis(100)) => { println!(..) }
์select!
์ ์ถ๊ฐํด ๋ณด์ธ์. ์ด ์์ ์ ์คํ๋์ง ์์ต๋๋ค. ์ ๊ทธ๋ด๊น์? -
๋์ , ํด๋น future๊ฐ ํฌํจ๋
timeout_fut
๋ฅผloop
์ธ๋ถ์ ์ถ๊ฐํด ๋ณด์ธ์.#![allow(unused)] fn main() { let mut timeout_fut = sleep(Duration::from_millis(100)); loop { select! { .., _ = timeout_fut => { println!(..); }, } } }
-
์ฌ์ ํ ์๋ํ์ง ์์ต๋๋ค. ์ปดํ์ผ๋ฌ ์ค๋ฅ๋ฅผ ๋ฐ๋ผ
select!
์timeout_fut
์&mut
๋ฅผ ์ถ๊ฐํ์ฌ Move ์๋ฉํฑ ๊ด๋ จํ ๋ฌธ์ ๋ฅผ ํด๊ฒฐํ๊ณBox::pin
์ ์ฌ์ฉํ์ธ์.#![allow(unused)] fn main() { let mut timeout_fut = Box::pin(sleep(Duration::from_millis(100))); loop { select! { .., _ = &mut timeout_fut => { println!(..); }, } } }
-
์ด๋ ์ปดํ์ผ์ ๋์ง๋ง ํ์ ์์์ด ๋๋ฉด ๋งค๋ฒ ๋ฐ๋ณตํ ๋ ๋ง๋ค
Poll::Ready
๊ฐ ๋ฉ๋๋ค(์ตํฉ๋ future๊ฐ ๋์์ด ๋ ์ ์์). ํ์ ์์ ๋ ๋๋ง๋คtimeout_fut
๋ฅผ ๋ฆฌ์ ํ๋๋ก ์์ ํ์ธ์.
-
-
Box๋ ํ์ ํ ๋นํฉ๋๋ค. ๊ฒฝ์ฐ์ ๋ฐ๋ผ
std::pin::pin!
(์ต๊ทผ์์ผ ์์ ํ๋์์ผ๋ฉฐ ์ด์ ์ฝ๋๋tokio::pin!
์ ์ฌ์ฉํ๋ ๊ฒฝ์ฐ๊ฐ ๋ง์)๋ ์ฌ์ฉํ ์ ์์ง๋ง ์ด๋ ์ฌํ ๋น๋ future์ ์ฌ์ฉํ๊ธฐ๊ฐ ์ด๋ ต์ต๋๋ค. -
๋ ๋ค๋ฅธ ๋ฐฉ๋ฒ์
pin
์ ์์ ์ฌ์ฉํ์ง ์๊ณ 100ms๋ง๋คoneshot
์ฑ๋์ ์ ์กํ ๋ค๋ฅธ ์์ ์ ์์ฑํ๋ ๊ฒ์ ๋๋ค.