Rust의 비동기 프로그래밍 소개
전통적인 동기식 프로그래밍 방식은 때때로 성능 저하의 원인이 됩니다. 이는 프로그램이 다음 단계로 넘어가기 전에 시간 소모적인 작업이 완료되기를 기다려야 하기 때문입니다. 이러한 대기 시간은 리소스 활용률을 떨어뜨리고, 사용자 경험을 느리게 만드는 주범이 됩니다.
반면, 비동기 프로그래밍은 시스템 자원을 효과적으로 활용하는 코드를 만들 수 있게 해줍니다. 비동기 프로그래밍을 사용하면 여러 작업을 동시에 처리하는 애플리케이션을 설계할 수 있습니다. 특히, 실행 흐름을 막지 않으면서 다양한 네트워크 요청을 처리하거나 방대한 데이터를 다룰 때 비동기 프로그래밍은 매우 유용합니다.
Rust에서의 비동기 프로그래밍
Rust의 비동기 프로그래밍 모델은 실행 흐름을 중단시키지 않고 병렬로 실행되는 효율적인 코드를 작성할 수 있도록 지원합니다. 비동기 프로그래밍은 특히 I/O 작업, 네트워크 요청 및 외부 리소스 대기와 관련된 작업에 적합합니다.
Rust 애플리케이션에서 비동기 프로그래밍을 구현하는 방법은 다양합니다. 여기에는 언어 자체 기능, 라이브러리, 그리고 Tokio 런타임 등이 포함됩니다.
더불어 Rust의 소유권 모델과 채널, 락 같은 동시성 기본 요소들은 안전하고 효율적인 동시 프로그래밍을 가능하게 합니다. 비동기 프로그래밍과 이러한 요소들을 결합하면 여러 CPU 코어를 효과적으로 활용하는 동시 시스템을 구축할 수 있습니다.
Rust 비동기 프로그래밍의 핵심 개념
Rust에서 비동기 프로그래밍의 토대는 Futures입니다. Future는 아직 완전히 실행되지 않은 비동기 계산을 나타냅니다.
Futures는 '느긋하게' 실행됩니다(즉, 폴링될 때만 실행됩니다). Future의 `poll()` 메서드를 호출하면 Future가 완료되었는지, 아니면 추가 작업이 필요한지 확인합니다. 만약 Future가 준비되지 않았다면 `Poll::Pending`을 반환하여 나중에 실행되어야 함을 알립니다. Future가 준비되면 결과 값과 함께 `Poll::Ready`를 반환합니다.
Rust 표준 툴체인은 비동기 I/O 프리미티브, 비동기 파일 I/O, 네트워킹, 그리고 타이머를 포함합니다. 이러한 요소들을 통해 I/O 작업을 비동기적으로 처리할 수 있습니다. 이를 통해 I/O 작업이 완료될 때까지 프로그램 실행이 중단되는 상황을 방지할 수 있습니다.
`async`/`await` 구문을 사용하면 마치 동기 코드처럼 보이는 비동기 코드를 작성할 수 있습니다. 이는 코드의 직관성을 높이고 유지 관리를 용이하게 만듭니다.

Rust의 비동기 프로그래밍 접근 방식은 안전성과 성능을 중요하게 생각합니다. 소유권 및 빌림 규칙은 메모리 안전을 보장하고 흔한 동시성 문제를 예방합니다. `async`/`await` 구문과 Future는 비동기 워크플로우를 표현하는 직관적인 방법을 제공합니다. 효율적인 실행을 위해 서드파티 런타임을 사용하여 작업을 관리할 수 있습니다.
이러한 언어 기능, 라이브러리, 그리고 런타임을 결합하여 고성능 코드를 작성할 수 있습니다. 이는 비동기 시스템을 구축하기 위한 강력하고 사용하기 편리한 프레임워크를 제공합니다. 결과적으로 Rust는 I/O 바운드 작업을 효율적으로 처리하고 높은 동시성을 요구하는 프로젝트에서 널리 사용됩니다.
Rust 1.39 버전 및 이후 릴리즈는 표준 라이브러리에서 비동기 작업을 직접 지원하지 않습니다. Rust에서 `async`/`await` 구문을 사용하여 비동기 작업을 처리하려면 서드파티 크레이트가 필요합니다. Tokio나 `async-std`와 같은 서드파티 패키지를 활용하여 `async`/`await` 구문을 사용할 수 있습니다.
Tokio를 이용한 비동기 프로그래밍
Tokio는 Rust를 위한 강력한 비동기 런타임입니다. 고성능 및 확장 가능한 애플리케이션을 구축하기 위한 다양한 기능을 제공합니다. Tokio를 사용하면 비동기 프로그래밍의 장점을 최대한 활용할 수 있으며, 확장성 또한 확보할 수 있습니다.
Tokio의 핵심은 비동기 작업을 예약하고 실행하는 모델입니다. Tokio를 사용하면 `async`/`await` 구문으로 비동기 코드를 작성할 수 있습니다. 이를 통해 시스템 리소스를 효율적으로 활용하고 동시에 여러 작업을 실행할 수 있습니다. Tokio의 이벤트 루프는 작업 스케줄링을 효율적으로 관리하며, CPU 코어를 최대한 활용하고 컨텍스트 전환 오버헤드를 줄여줍니다.
Tokio의 조합자(combinators)는 작업 조정 및 구성 작업을 간편하게 만듭니다. Tokio는 여러 작업을 조정하고 구성하는 강력한 도구를 제공합니다. `join`을 사용하면 여러 작업이 완료될 때까지 기다릴 수 있으며, `select`를 사용하면 완료된 첫 번째 작업을 선택할 수 있고, `race`를 사용하면 여러 작업을 서로 경쟁시킬 수 있습니다.
`Cargo.toml` 파일의 의존성 섹션에 Tokio 크레이트를 추가합니다.
[dependencies]
tokio = { version = "1.9", features = ["full"] }
다음은 Tokio를 사용하여 Rust 프로그램에서 `async`/`await` 구문을 사용하는 방법의 예시입니다.
use tokio::time::sleep;
use std::time::Duration;async fn hello_world() {
println!("Hello, ");
sleep(Duration::from_secs(1)).await;
println!("World!");
}#[tokio::main]
async fn main() {
hello_world().await;
}
`hello_world` 함수는 비동기 함수이므로, `await` 키워드를 사용하여 Future가 해결될 때까지 실행을 일시 중지할 수 있습니다. `hello_world` 함수는 먼저 "Hello,"를 콘솔에 출력합니다. `Duration::from_secs(1)` 함수 호출은 함수 실행을 잠시 멈춥니다. `await` 키워드는 sleep Future가 완료될 때까지 기다립니다. 마지막으로, `hello_world` 함수는 "World!"를 콘솔에 출력합니다.

`main` 함수에는 `#[tokio::main]` 어트리뷰트가 적용되어 있습니다. 이는 Tokio 런타임의 진입점으로 사용될 메인 함수를 지정합니다. `hello_world().await;`는 `hello_world` 함수를 비동기적으로 실행합니다.
Tokio를 사용한 작업 지연
비동기 프로그래밍에서 흔히 사용되는 작업 중 하나는 지연을 사용하거나 특정 시간 간격으로 실행되도록 작업을 예약하는 것입니다. Tokio 런타임은 `tokio::time` 모듈을 통해 비동기 타이머와 지연을 사용하는 메커니즘을 제공합니다.
Tokio 런타임으로 작업을 지연하는 방법의 예시는 다음과 같습니다.
use std::time::Duration;
use tokio::time::sleep;async fn delayed_operation() {
println!("Performing delayed operation...");
sleep(Duration::from_secs(2)).await;
println!("Delayed operation completed.");
}#[tokio::main]
async fn main() {
println!("Starting...");
delayed_operation().await;
println!("Finished.");
}
`delayed_operation` 함수는 `sleep` 메서드를 사용하여 2초 동안 지연을 발생시킵니다. `delayed_operation` 함수는 비동기 함수이므로, `await`를 사용하여 지연이 완료될 때까지 실행을 일시 중단할 수 있습니다.

비동기 프로그램의 오류 처리
비동기 Rust 코드에서 오류를 처리할 때는 `Result` 타입을 사용하고 `?` 연산자를 활용합니다.
use tokio::fs::File;
use tokio::io;
use tokio::io::{AsyncReadExt};async fn read_file_contents() -> io::Result<String> {
let mut file = File::open("file.txt").await?;
let mut contents = String::new();
file.read_to_string(&mut contents).await?;
Ok(contents)
}async fn process_file() -> io::Result<()> {
let contents = read_file_contents().await?;
Ok(())
}#[tokio::main]
async fn main() {
match process_file().await {
Ok(()) => println!("File processed successfully."),
Err(err) => eprintln!("Error processing file: {}", err),
}
}
`read_file_contents` 함수는 I/O 오류 가능성을 나타내는 `io::Result`를 반환합니다. 각 비동기 작업 후 `?` 연산자를 사용하여 Tokio 런타임이 오류를 호출 스택 위로 전파할 수 있게 합니다.
`main` 함수는 작업 결과를 처리하기 위해 `match` 문을 사용합니다. 결과에 따라 다른 메시지를 출력합니다.
Reqwest를 이용한 비동기 HTTP 작업
Reqwest를 비롯한 많은 인기 크레이트들은 Tokio를 사용하여 비동기 HTTP 작업을 제공합니다.
Reqwest와 Tokio를 함께 사용하면 다른 작업을 막지 않으면서 여러 HTTP 요청을 처리할 수 있습니다. Tokio는 수천 개의 동시 연결을 처리하고 리소스를 효율적으로 관리하는 데 도움을 줄 수 있습니다.