Что такое tokio::time::sleep, timeout и interval? Как их использовать?
sleep задерживает задачу без блокировки потока, timeout оборачивает future с дедлайном, interval генерирует периодические тики (первый — немедленно). MissedTickBehavior::Skip рекомендован для production, чтобы избежать Burst после пауз.
tokio::time: sleep, timeout, interval
Модуль tokio::time предоставляет три основных инструмента для работы со временем в асинхронном коде. Все они работают без блокировки потока — вместо std::thread::sleep runtime просто «засыпает» текущую задачу и выполняет другие.
tokio::time::sleep — одноразовая задержка
use tokio::time::{sleep, Duration};
#[tokio::main]
async fn main() {
println!("start");
sleep(Duration::from_millis(500)).await;
println!("after 500ms");
// sleep_until принимает Instant
use tokio::time::{sleep_until, Instant};
let deadline = Instant::now() + Duration::from_secs(1);
sleep_until(deadline).await;
println!("after absolute deadline");
}
tokio::time::Instant — монотонные часы, изолированные от системных (можно паузировать в тестах). Не путайте с std::time::Instant.
tokio::time::timeout — ограничение времени выполнения
use tokio::time::{timeout, Duration};
use tokio::net::TcpStream;
async fn connect_with_timeout(addr: &str) -> anyhow::Result<TcpStream> {
match timeout(Duration::from_secs(5), TcpStream::connect(addr)).await {
Ok(Ok(stream)) => Ok(stream),
Ok(Err(e)) => Err(e.into()), // соединение ошибка
Err(_elapsed) => Err(anyhow::anyhow!("connection timed out")),
}
}
timeout оборачивает любую future. Если она не завершилась за отведённое время, возвращается Err(Elapsed). Внутренняя future отменяется (дропается) при истечении таймаута — убедитесь в её cancellation safety.
timeout_at — с абсолютным дедлайном
use tokio::time::{timeout_at, Instant, Duration};
async fn handle_request_with_budget(deadline: Instant) {
// несколько операций делят общий бюджет времени
let _ = timeout_at(deadline, step_one()).await;
let _ = timeout_at(deadline, step_two()).await;
}
tokio::time::interval — периодические задачи
use tokio::time::{interval, Duration};
#[tokio::main]
async fn main() {
let mut ticker = interval(Duration::from_secs(1));
for i in 0..5 {
ticker.tick().await; // первый tick срабатывает немедленно!
println!("tick {i}");
}
}
Важно: первый tick() срабатывает сразу же при создании интервала. Если нужно начать с задержки, создайте интервал через interval_at:
use tokio::time::{interval_at, Instant, Duration};
let start = Instant::now() + Duration::from_secs(5);
let mut ticker = interval_at(start, Duration::from_secs(1));
ticker.tick().await; // первый tick через 5 секунд
MissedTickBehavior — что делать с пропущенными тиками
use tokio::time::{interval, Duration, MissedTickBehavior};
let mut ticker = interval(Duration::from_secs(1));
ticker.set_missed_tick_behavior(MissedTickBehavior::Skip);
// Burst — немедленно возвращает все пропущенные тики (по умолчанию)
// Skip — пропускает пропущенные, следующий тик через period от текущего момента
// Delay — откладывает следующий тик на period от текущего момента
Практический пример: healthcheck loop
use tokio::time::{interval, timeout, Duration};
use tokio::sync::broadcast;
async fn healthcheck_loop(
url: String,
mut shutdown_rx: broadcast::Receiver<()>,
) {
let mut ticker = interval(Duration::from_secs(30));
ticker.set_missed_tick_behavior(tokio::time::MissedTickBehavior::Skip);
loop {
tokio::select! {
_ = ticker.tick() => {
match timeout(Duration::from_secs(5), ping(&url)).await {
Ok(Ok(())) => println!("healthy"),
Ok(Err(e)) => eprintln!("unhealthy: {e}"),
Err(_) => eprintln!("probe timed out"),
}
}
_ = shutdown_rx.recv() => break,
}
}
}
async fn ping(_url: &str) -> anyhow::Result<()> { Ok(()) }
Подводные камни
- Первый
interval.tick()срабатывает немедленно — это удивляет большинство разработчиков. Используйтеinterval_at(Instant::now() + period, period)для отложенного старта. - Поведение по умолчанию
MissedTickBehavior::Burstможет вызвать всплеск задач после паузы (например, после GC или нагрузки). Для большинства production-задач лучше подходитSkip. timeoutотменяет внутреннюю future при истечении времени — если future не cancellation-safe, данные могут быть потеряны.tokio::time::Instantиstd::time::Instantнесовместимы — нельзя передавать одно вместо другого.- В тестах время Tokio можно паузировать с помощью
tokio::time::pause()и перематывать черезtokio::time::advance()— не используйтеstd::thread::sleepв async тестах, это блокирует поток. - Если runtime «занят» (CPU-bound задачи без yield), таймеры не срабатывают вовремя — time-based гарантии в Tokio мягкие (best-effort).
- Забытый
mutу переменной interval (let tickerвместоlet mut ticker) — компилятор поймает, но это частая опечатка при быстром написании.
Common mistakes
- Отвечать определением без production-сценария.
- Не называть runtime boundary, security boundary или failure mode.
- Игнорировать версию API, observability и тестовую проверку.
What the interviewer is testing
- Объясняет механизм своими словами и без выдуманных API.
- Называет реальные риски, диагностику и критерий корректности.
- Связывает ответ с текущей документацией и миграционными ограничениями.