Comprehensive Rust์ ์ค์ ๊ฒ์ ํ์ํฉ๋๋ค ๐ฆ
์ด ๊ฐ์๋ ๋ฌด๋ฃ์ด๋ฉฐ, Google์ Android ํ์ด ๋ง๋ค์์ต๋๋ค. ๊ธฐ๋ณธ ๋ฌธ๋ฒ๋ถํฐ ์ ๋ค๋ฆญ, ์๋ฌ ํธ๋ค๋ง๊ณผ ๊ฐ์ ๊ณ ๊ธ์ฃผ์ ๊น์ง ๋ฌ์คํธ์ ๋ชจ๋ ๊ฒ์ ํฌํจํฉ๋๋ค.
The latest version of the course can be found at https://google.github.io/comprehensive-rust/. If you are reading somewhere else, please check there for updates.
๊ฐ์๋ ๋น์ ์ด ๋ฌ์คํธ์ ๋ํด์ ์๋ฌด๊ฒ๋ ๋ชจ๋ฅธ๋ค๊ณ ๊ฐ์ ํ๊ณ ์๋์ ๋ชฉํ๋ฅผ ๊ฐ์ง๊ณ ์์ต๋๋ค:
- ๋ฌ์คํธ ๊ตฌ๋ฌธ๊ณผ ์ธ์ด์ ๋ํ ํฌ๊ด์ ์ธ ์ดํด๋ฅผ ์ ๊ณตํฉ๋๋ค.
- ๊ธฐ์กด ํ๋ก๊ทธ๋จ์ ์์ ํ๊ณ ๋ฌ์คํธ์์ ์ ํ๋ก๊ทธ๋จ์ ์์ฑํ ์ ์์ต๋๋ค.
- ์ผ๋ฐ์ ์ธ ๋ฌ์คํธ ๊ด์ฉ๊ตฌ๋ฅผ ๋ณด์ฌ์ค๋๋ค.
๊ฐ์์ ์ฒซ 3์ผ ๋์์๋ Rust์ ๊ธฐ์ด๋ฅผ ๋ค์ง๋๋ค.
๊ทธ ํ์๋, ์๋์ ๊ฐ์ ๊ฐ๋ณ ์ฃผ์ ๋ฅผ ์ฌํํด์ ๊ณต๋ถํ ์ ์์ต๋๋ค:
- Android: Android ํ๋ซํผ ๊ฐ๋ฐ(AOSP) ์ Rust ์ฌ์ฉ์ ๊ดํ ๋ฐ๋์ ๊ณผ์ ์ ๋๋ค. ์ฌ๊ธฐ์๋ C, C++, Java์์ ์ํธ ์ด์ฉ์ฑ์ด ํฌํจ๋ฉ๋๋ค.
- Bare-metal: bare-metal(์๋ฒ ๋๋) ๊ฐ๋ฐ ์ Rust ์ฌ์ฉ์ ๊ดํ ์ข ์ผ ๊ณผ์ ์ ๋๋ค. ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์ ์ ํ๋ฆฌ์ผ์ด์ ํ๋ก์ธ์๋ฅผ ๋ชจ๋ ๋ค๋ฃน๋๋ค.
- ๋์์ฑ: Rust์ ๋์์ฑ์ ๊ดํ ์ข ์ผ ๊ณผ์ ์ ๋๋ค. ์ฌ๊ธฐ์๋ ๊ณ ์ ์ ์ธ ๋์์ฑ(์ค๋ ๋์ ๋ฎคํ ์ค๋ฅผ ์ฌ์ฉํ์ฌ ์ ์ ํ ์ค์ผ์ค๋ง์ ํ๋ ๊ฒ)๊ณผ async/await ๋์์ฑ(future๋ฅผ ์ฌ์ฉํ๋ ํ๋ ฅ์ ์ธ ๋ฉํฐํ์คํน)์ ๋ชจ๋ ๋ค๋ฃน๋๋ค.
์ ์ธ์ฌํญ
๋ฌ์คํธ๋ ๋ฉฐ์น ๋ง์ ๋ชจ๋ ๊ฒ์ ๋ค๋ฃจ๊ธฐ์๋ ๋๋ฌด ํฐ ์ธ์ด์ ๋๋ค. ๊ทธ๋์ ์๋์ ๊ฐ์๊ฒ์ ๋ชฉํ๋ก ํ์ง ์์ต๋๋ค:
- ๋งคํฌ๋ก ๋ง๋ค๊ธฐ: ๋งคํฌ๋ก์ ๊ด๋ จํ ์์ธํ ๋ด์ฉ์ ๋ฌ์คํธ ํ๋ก๊ทธ๋๋ฐ ์ธ์ด, 19.1์ ๊ณผ Rustonomicon๋ฅผ ์ฐธ์กฐํ์ธ์.
๋ ์ ์์ค์ ๋ํ ๊ฐ์
๋ณธ ๊ฐ์๋ ์ฌ๋ฌ๋ถ์ด ํ๋ก๊ทธ๋๋ฐ ์์ฒด์ ๋ํด์๋ ์๊ณ ์๋ค๊ณ ๊ฐ์ ํฉ๋๋ค. ๋ฌ์คํธ๋ ์ ์ ํ์ ์ธ์ด์ด๋ฉฐ, ๊ฐ์ข์์๋ C/C++ ์์ ๋น๊ต, ๋์กฐ๋ฅผ ํตํด ๋ฌ์คํธ๋ฅผ ์ค๋ช ํ ๊ฒ์ ๋๋ค.
C/C++์ ๋ชจ๋ฅด๋๋ผ๋ ๋์ ํ์ ์ธ์ด(Python์ด๋ JavaScript ๋ฑ) ํ๋ก๊ทธ๋๋ฐ ๊ฒฝํ์ด ์๋ค๋ฉด ๋ฐ๋ผ์ค๋๋ฐ ํฐ ๋ฌธ์ ๋ ์์ ๊ฒ์ ๋๋ค.
์ด๊ฒ์ โ๋ฐํ์ ๋ ธํธโ์ ์์ ์ ๋๋ค. ์ด ๋ถ๋ถ์ ์ด์ฉํด์ ์ถ๊ฐ ์ ๋ณด๋ฅผ ์ ๊ณตํฉ๋๋ค. ์ฃผ๋ก ๊ฐ์์ค์์ ์ ๊ธฐ๋๋ ์ผ๋ฐ์ ์ธ ์ง๋ฌธ์ ๋ํ ๋ต๋ณ๊ณผ ๊ฐ์ฌ๊ฐ ๋ค๋ฃจ์ด์ผ ํ ํค ํฌ์ธํธ์ผ ์ ์์ต๋๋ค.
๊ฐ์ ์งํ
๊ฐ์ฌ๋ฅผ ์ํ ์๋ด ํ์ด์ง์ ๋๋ค.
๋ค์์ ๊ตฌ๊ธ ๋ด๋ถ์์ ์ด ๊ณผ์ ์ ์ด๋ค์์ผ๋ก ์ด์ํด์๋์ง์ ๋ํ ๋ฐฐ๊ฒฝ ์ ๋ณด์ ๋๋ค.
We typically run classes from 10:00 am to 4:00 pm, with a 1 hour lunch break in the middle. This leaves 2.5 hours for the morning class and 2.5 hours for the afternoon class. Note that this is just a recommendation: you can also spend 3 hour on the morning session to give people more time for exercises. The downside of longer session is that people can become very tired after 6 full hours of class in the afternoon.
๊ฐ์๋ฅผ ์คํํ๊ธฐ ์ํ ์ค๋น:
-
๊ฐ์ ์๋ฃ๋ฅผ ์์งํฉ๋๋ค. ์ฃผ์ ์์ ์ ๊ฐ์กฐํ๊ธฐ ์ํด ๊ฐ์ ์ฐธ์กฐ ๋ ธํธ๋ฅผ ํฌํจํ์์ต๋๋ค. (์ถ๊ฐ์ ์ธ ๋ ธํธ๋ฅผ ์์ฑํ์ฌ ์ ๊ณตํด ์ฃผ์๋ฉด ๊ฐ์ฌํ๊ฒ ์ต๋๋ค.) ๊ฐ์ ์ฐธ์กฐ ๋ ธํธ์ ๋งํฌ๋ฅผ ๋๋ฅด๋ฉด ๋ณ๋์ ํ์ ์ผ๋ก ๋ถ๋ฆฌ๊ฐ ๋๋ฉฐ, ๋ฉ์ธ ํ๋ฉด์์๋ ์ฌ๋ผ์ง๋๋ค. ๊น๋ํ ํ๋ฉด์ผ๋ก ๊ฐ์๋ฅผ ์งํํ ์ ์์ต๋๋ค.
-
๊ฐ์ ๋ ์ง๋ฅผ ์ ํฉ๋๋ค. ์ต์ 3์ผ์ ๊ฑธ์ณ์ ์งํ์ด ๋๊ธฐ ๋๋ฌธ์, ๋ ์ฃผ์ ๊ฑธ์ณ์ ์ค์ผ์ค์ ์ก๋ ๊ฒ์ ๊ถํฉ๋๋ค. ๊ธฐ์กด ๊ฐ์ ์๊ฐ์๋ค์ ํผ๋๋ฐฑ์ ๋ฐ๋ฅด๋ฉด, ์ฐ๋ฌ์์ ๊ฐ์๋ฅผ ์งํํ๋ ๊ฒ ๋ณด๋ค, ๊ฐ์๋ฅผ ๋์๋์ ํ๋ ๊ฒ์ด ๊ฐ์ ๋ด์ฉ์ ์ํํ๋๋ฐ ๋ ๋์์ด ๋์๋ค๊ณ ํฉ๋๋ค.
-
์ถฉ๋ถํ ๊ณต๊ฐ์ ํ๋ณดํฉ๋๋ค. 15์์ 20๋ช ๊ท๋ชจ์ ๊ณต๊ฐ์ ์ถ์ฒํฉ๋๋ค. ์๊ฐ์๊ณผ ๊ฐ์ฌ๊ฐ ์ง์๋ฅผ ํ๊ธฐ์ ์ถฉ๋ถํ ์๊ฐ๊ณผ ๊ณต๊ฐ์ด์ด์ผ ํฉ๋๋ค. ๊ฐ์ฌ๋ ์๊ฐ์ ๋ชจ๋ _์ฑ ์_์ ์ฌ์ฉํ ์ ์๋ ๊ฐ์์ค์ด๋ฉด ์ข์ต๋๋ค. ๊ฐ์ ์ค์ ๊ฐ์ฌ๊ฐ ๋ผ์ด๋ธ ์ฝ๋ฉ์ ํ๊ฒ ๋ ๊ฒฝ์ฐ๊ฐ ๋ง์ผ๋ฉฐ, ์ด๋ ์๋ฆฌ์ ์์ ๋ ธํธ๋ถ์ ์ฌ์ฉํ๋ ๊ฒ์ด ๋์์ด ๋ฉ๋๋ค.
-
๊ฐ์ ๋น์ผ ์กฐ๊ธ ์ผ์ฐ ์์ ์ค๋นํฉ๋๋ค. ๊ฐ์ฌ ๋ ธํธ๋ถ์์
mdbook serve
๋ฅผ ์ด์ฉํด ์ง์ ํ๋ ์ ํ ์ด์ ํ๋ฉด ํ์ด์ง ์ด๋ ์์ ์ง์ฐ์ด ์์ต๋๋ค.(์ค์น ๋ฐฉ๋ฒ์ ์ฐธ์กฐํ์ธ์.) ๋ํ, ๊ทธ๋ ๊ฒ ํ๋ฉด ๊ฐ์ ๋์ค ์คํ๋ฅผ ๋ฐ๊ฒฌํ์ ๋ ๊ทธ ์๋ฆฌ์์ ๋ฐ๋ก ์์ ๊ฐ๋ฅํ๋ค๋ ์ฅ์ ๋ ์์ต๋๋ค. -
์๊ฐ์๋ค์ด ์ง์ (๊ฐ๋ณ ํน์ ๊ทธ๋ฃน์ผ๋ก) ์ฐ์ต๋ฌธ์ ๋ฅผ ํ๋๋ก ํฉ๋๋ค. ๋์ฒด๋ก ์ค์ , ์คํ์ ๊ฐ๊ฐ 30-45๋ถ ์ ๋๋ฅผ ์ฐ์ต๋ฌธ์ ์ ํ ๋นํฉ๋๋ค (์ด๋ ํด๋ต์ ๋ณด๊ณ ์ค๋ช ํ๋ ์๊ฐ๊น์ง ํฌํจํฉ๋๋ค). ๋งํ ๋์์ ํ์๋ก ํ๋ ์๊ฐ์์ด ์๋์ง ์์๋ก ํ์ธํฉ๋๋ค. ๋ง์ฝ ๊ฐ์ ๋ฌธ์ ๋ฅผ ์ฌ๋ฌ ์ฌ๋์ด ๊ฒช๊ณ ์๋ค๋ฉด, ๊ทธ ๋ฌธ์ ๋ฅผ ๊ฐ์์ค ์ ์ฒด ์ธ์์๊ฒ ์๋ฆฌ๊ณ ํด๊ฒฐ์ฑ ์ ์ ์ํฉ๋๋ค. ์๋ฅผ ๋ค์ด ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ ์ด๋์ ๊ฐ๋ฉด ๊ทธ ๋ฌธ์ ์ ๋ํ ํด๋ต์ ์ฐพ์ ์ ์๋์ง ์๋ ค ์ค๋๋ค.
์ด์ ์ค๋น๋ ๋๋ฌ์ต๋๋ค. ์ฐ๋ฆฌ๊ฐ ๊ทธ๋ฌ๋ฏ์ด ์ฌ๋ฌ๋ถ๋ค๋ ์ด ๊ฐ์๋ฅผ ์ฆ๊ธฐ์๊ธธ ๋ฐ๋๋๋ค!
๊ฐ์๋ฅผ ๊ณ์ ๊ฐ์ ํ ์ ์๋๋ก ํผ๋๋ฐฑ์ ์ ๊ณตํด ์ฃผ์ญ์์ค. ์ฐ๋ฆฌ๋ ๋ฌด์์ด ์ข์๊ณ , ๋ฌด์์ด ๋ชจ์๋๋์ง ๋ฃ๊ณ ์ถ์ต๋๋ค. ์๊ฐ์๋ค๋ก ๋ถํฐ์ ํผ๋๋ฐฑ๋ ํ์ํฉ๋๋ค!
๊ฐ์ ๊ตฌ์ฑ
๊ฐ์ฌ๋ฅผ ์ํ ์๋ด ํ์ด์ง์ ๋๋ค.
Rust Fundamentals
The first three days make up Rust Fundaments. The days are fast paced and we cover a lot of ground:
- Day 1: Basic Rust, syntax, control flow, creating and consuming values.
- Day 2: Memory management, ownership, compound data types, and the standard library.
- Day 3: Generics, traits, error handling, testing, and unsafe Rust.
์ฌํ ํ์ต
Rust ๊ธฐ์ด์ ๊ดํ 3์ผ ๊ณผ์ ์ดํ์๋, ๋ค์๊ณผ ๊ฐ์ ์ ๋ฌธ ์ฃผ์ ๋ฅผ ๋ค๋ฃน๋๋ค:
Rust in Android
The Rust in Android deep dive is a half-day course on using Rust for Android platform development. This includes interoperability with C, C++, and Java.
AOSP ์ฝ๋๋ฅผ ์ฌ๋ฌ๋ถ์ ์ปดํจํฐ์ ์ฒดํฌ์์ํด์ผ ํฉ๋๋ค. ๊ทธ๋ฐ ๋ค์, ๊ทธ ์ปดํจํฐ์์ ๊ณผ์ ์ ์ฅ์๋ฅผ ์ฒดํฌ์์ํ๊ณ src/android/
๋๋ ํฐ๋ฆฌ๋ฅผ AOSP ์ฝ๋์ ๋ฃจํธ๋ก ์ด๋ํฉ๋๋ค. ์ด๋ ๊ฒ ํ๋ฉด ์๋๋ก์ด๋ ๋น๋ ์์คํ
์์ ๊ณผ์ ์ฉ์ผ๋ก ์ถ๊ฐ๋ Android.bp
ํ์ผ์ ์ธ์ํ ์ ์์ต๋๋ค.
adb sync
๋ช
๋ ์ด๊ฐ ์๋ฎฌ๋ ์ดํฐ ํน์ ์ค์ ์ฅ์น์ ์๋ํ๋์ง ํ์ธํฉ๋๋ค. ๊ทธ๋ฆฌ๊ณ src/android/build_all.sh
๋ฅผ ์ํํด์ ๋ชจ๋ ์๋๋ก์ด๋ ์์ ๋ฅผ ๋ฏธ๋ฆฌ ๋น๋ํด ๋ณด์ธ์. ๊ทธ ์ ์คํฌ๋ฆฝํธ๋ฅผ ์ฝ๊ณ , ๊ทธ ์์์ ์ํ๋๋ ๋ช
๋ น์ด๋ค์ ํ์ธํ ํ ๊ฐ ๋ช
๋ น์ด๋ค์ ์๋์ผ๋ก ์คํํด๋ ์ ๋๋์ง ํ์ธํ์ธ์.
Bare-Metal Rust
The Bare-Metal Rust deep dive is a full day class on using Rust for bare-metal (embedded) development. Both microcontrollers and application processors are covered.
๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ ํํธ๋ฅผ ์งํํ๊ธฐ ์ํด์๋ BBC micro:bit v2 ๊ฐ๋ฐ ๋ณด๋๋ฅผ ๋ฏธ๋ฆฌ ๊ตฌ๋งคํด์ผ ํฉ๋๋ค. ๋ชจ๋ ์ฌ์ฉ์๋ ์์ ํ์ด์ง์ ์ค๋ช ๋ ๋๋ก ๊ฐ์ข ํจํค์ง๋ฅผ ์ค์นํด์ผ ํฉ๋๋ค.
Concurrency in Rust
The Concurrency in Rust deep dive is a full day class on classical as well as async
/await
concurrency.
์ ํฌ๋ ์ดํธ๋ฅผ ์ค์ ํ๊ณ ๋ช ๊ฐ์ง ์์กด์ฑ์ ๋ค์ด๋ก๋ํด ๋์ด์ผ ํฉ๋๋ค. ๊ทธ๋ฐ ๋ค์ ์์ ๋ฅผ src/main.rs
์ ๋ณต์ฌ/๋ถ์ฌ๋ฃ๊ธฐ ํ์ฌ ํ
์คํธ ํด ๋ณผ ์ ์์ต๋๋ค:
cargo init concurrency
cd concurrency
cargo add tokio --features full
cargo run
๊ฐ์ ํ์
์ด ๊ฐ์๋ ๊ฐ์ฌ์ ์๊ฐ์์ด ์๋ฐฉํฅ์ผ๋ก ์ํตํ๋ฉด์ ์งํํ๋๋ก ๋์์ธ ๋์์ต๋๋ค. ๋ค์ํ ์ง๋ฌธ์ ํตํด ๋ฌ์คํธ์ ์ฌ๋ฌ ๋ถ๋ถ์ ํํํ ์ ์๋๋ก ํ์ธ์!
๋จ์ถํค
๋ค์์, mdBook ์์คํ (ํ ์ฌ์ดํธ)์์ ์ ์ฉํ ๋จ์ถํค๋ค ์ ๋๋ค:
- ์ผ์ชฝ ํ์ดํ: ์ด์ ํ์ด์ง๋ก ์ด๋ํฉ๋๋ค.
- ์ค๋ฅธ์ชฝ ํ์ดํ: ๋ค์ ํ์ด์ง๋ก ์ด๋ํฉ๋๋ค.
- Ctrl + Enter: ํ์ฌ ํฌ์ปค์ค๋ฅผ ๋ฐ์ ์ฝ๋ ์ํ ๋ธ๋ก์ ์คํํฉ๋๋ค.
- s: ๊ฒ์์ฐฝ์ ํ์ฑํํฉ๋๋ค.(mdBook ๋ฌธ์ ๋ก 23.01.19 ๊ธฐ์ค ์์ด๋ก๋ง ๊ฐ๋ฅํฉ๋๋ค.)
๋ค๋ฅธ ์ธ์ด๋ค
์ด ๊ณผ์ ์ ๋ค๋ฅธ ์ธ์ด๋ก๋ ์ ๊ณต๋ฉ๋๋ค. ๊ดํธ ์์ ๋ฒ์ญ์ ๋์ ์ฃผ์ ๋ถ๋ค์ ๋๋ค:
- Brazilian Portuguese by @rastringer, @hugojacob, @joaovicmendes, and @henrif75.
- Korean by @keispace, @jiyongp, and @jooyunghan.
- Spanish by @deavid.
ํ์ด์ง ์ค๋ฅธ์ชฝ ์์ ๋ฉ๋ด๋ฅผ ํตํด ๋ค๋ฅธ ์ธ์ด๋ก ์ ํํ ์ ์์ต๋๋ค.
๋ฒ์ญ ๋ฌธ์
์งํ ์ค์ธ ๋ฒ์ญ์ด ๋ง์ต๋๋ค. ์ต๊ทผ์ ์ ๋ฐ์ดํธ๋ ๋ฒ์ญ๋ณธ์ผ๋ก ์ฐ๊ฒฐ๋๋ ๋งํฌ๋ ๋ค์๊ณผ ๊ฐ์ต๋๋ค:
- ๋ฒต๊ฐ์ด: @raselmandol ์ ๊ณต.
- Chinese (Traditional) by @hueich, @victorhsieh, @mingyc, and @johnathan79717.
- Chinese (Simplified) by @suetfei, @wnghl, @anlunx, @kongy, @noahdragon, and @superwhd.
- ํ๋์ค์ด: @KookaS ๋ฐ @vcaen ์ ๊ณต.
- ๋ ์ผ์ด: @Throvn ๋ฐ @ronaldfw ์ ๊ณต.
- ์ผ๋ณธ์ด: @CoinEZ-JPN ๋ฐ @momotaro1105 ์ ๊ณต.
์ด ๊ณผ์ ์ ๋ฒ์ญ ์์ ์ ๋์์ ์ฃผ๊ณ ์ถ๋ค๋ฉด ์ฌ๊ธฐ ์ค๋ช ๋ ๋ด์ฉ์ ์ฐธ๊ณ ํ์ธ์. ์งํ ์ค์ธ ๋ฒ์ญ ์์ ์ ๋ํ ๋ด์ฉ์ ์ด์ ํธ๋์ปค๋ฅผ ์ฐธ๊ณ ํ์ธ์.
์นด๊ณ ์ฌ์ฉํ๊ธฐ
๋ฌ์คํธ๋ฅผ ์์ํ๋ ค๊ณ ํ๋ฉด ๋น์ ์ ๊ณง Cargo๋ผ๋, ๋ฌ์คํธ ์ํ๊ณ์์ ์ฌ์ฉํ๋ ํ์ค ๋น๋/์คํ ๋๊ตฌ๋ฅผ ๋ง๋ ๊ฒ ์ ๋๋ค. ์ฌ๊ธฐ์๋ ์นด๊ณ ๊ฐ ๋ฌด์์ธ์ง, ๊ทธ๋ฆฌ๊ณ ์นด๊ณ ๊ฐ ๋ฌ์คํธ ์ํ๊ณ์์ ์ด๋ค ์ญํ ์ ํ๋์ง, ๊ทธ๋ฆฌ๊ณ ์ด ๊ฐ์์์ ์ด๋ป๊ฒ ์ฌ์ฉ๋ ์ง์ ๋ํด ๊ฐ๋ตํ ์ค๋ช ํ๊ฒ ์ต๋๋ค.
์ค์นํ๊ธฐ
https://rustup.rs/์ ์ค์น ๋ฐฉ๋ฒ์ ๋ฐ๋ฅด์ธ์.
This will give you the Cargo build tool (cargo
) and the Rust compiler (rustc
). You will also get rustup
, a command line utility that you can use to install to different compiler versions.
After installing Rust, you should configure your editor or IDE to work with Rust. Most editors do this by talking to rust-analyzer, which provides auto-completion and jump-to-definition functionality for VS Code, Emacs, Vim/Neovim, and many others. There is also a different IDE available called RustRover.
-
๋ฐ๋น์/์ฐ๋ถํฌ ์์คํ ์์๋
apt
๋ฅผ ์ด์ฉํด์ ์นด๊ณ , ๋ฌ์คํธ ์์ค, ๋ฌ์คํธ ํฌ๋งคํฐ๋ฅผ ์ค์นํ ์ ์์ต๋๋ค. ๊ทธ๋ฌ๋ ์ด ๋ฐฉ๋ฒ์ ๋ฐ๋ฅผ ๊ฒฝ์ฐ ์ต์ ๋ฒ์ ์ด ์๋ ๋ฌ์คํธ๋ฅผ ์ฌ์ฉ๊ฒ๋๋ฉฐ, ๊ทธ ๊ฒฐ๊ณผ ์์์น ๋ชปํ ๋ฌธ์ ๋ฅผ ๊ฒช์ ์๋ ์์ต๋๋ค. ์ค์น ๋ช ๋ น์ด๋ ์๋์ ๊ฐ์ต๋๋ค:sudo apt install cargo rust-src rustfmt
๋ฌ์คํธ ์ํ๊ณ
๋ฌ์คํธ์ ์ํ๊ณ๋ ์ฌ๋ฌ๊ฐ์ง ๋๊ตฌ๋ค๋ก ๊ตฌ์ฑ๋์ด ์์ผ๋ฉฐ, ๊ทธ ์ค ์ค์ํ ๊ฒ๋ค์ ์๋์ ๊ฐ์ต๋๋ค:
-
rustc
:.rs
ํ์ฅ์ ํ์ผ์ ๋ฐ์ด๋๋ฆฌ ํน์ ๋ค๋ฅธ ์ค๊ฐ ํ์์ผ๋ก ๋ณํํด์ฃผ๋ Rust ์ปดํ์ผ๋ฌ์ ๋๋ค. -
cargo
: ๋ฌ์คํธ ์์กด์ฑ ๊ด๋ฆฌ์์ด์ ๋น๋ ์์คํ ์ ๋๋ค. ์ฌ๋ฌ๋ถ์ ํ๋ก์ ํธ์ ๋ช ์๋ ์์กด์ฑ๋ค์ https://crates.io์์ ์๋์ผ๋ก ๋ค์ด๋ก๋ ๋ฐ๊ณ , ๊ทธ ์์ค์ฝ๋๋ฅผrustc
๋ก ์ ๋ฌํ์ฌ ๋น๋๋ฅผ ์ํต๋๋ค. ๋ํ ์ ๋ ํ ์คํธ๋ฅผ ์คํํ๋ ํ ์คํธ ๋ฌ๋๋ฅผ ๋ด์ฅํ๊ณ ์์ต๋๋ค. -
rustup
: ๋ฌ์คํธ ํด์ฒด์ธ ์ค์น ํ๋ก๊ทธ๋จ์ด์ ์ ๋ฐ์ดํธ ํ๋ก๊ทธ๋จ ์ ๋๋ค. ์ด ๋๊ตฌ๋ ์ ๋ฒ์ ์ ๋ฌ์คํธ๊ฐ ์ถ์๋ ๋rustc
๋ฐcargo
์ค์นํ๊ณ ์ ๋ฐ์ดํธํ๋ ๋ฐ ์ฌ์ฉ๋ฉ๋๋ค. ๋ํrustup
์ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ๋ํ ๋ฌธ์๋ฅผ ๋ค์ด๋ก๋ํ ์๋ ์์ต๋๋ค. ํ ๋ฒ์ ์ฌ๋ฌ ๋ฒ์ ์ ๋ฌ์คํธ๋ฅผ ์ค์นํ ์ ์์ผ๋ฉฐrustup
์ ์ด์ฉํด์ ์ค์ ๋ก ์ฌ์ฉํ ๋ฒ์ ์ ๋ณ๊ฒฝํ ์ ์์ต๋๋ค.
ํค ํฌ์ธํธ:
-
๋ฌ์คํธ๋ 6์ฃผ๋ง๋ค ์๋ก์ด ๋ฆด๋ฆฌ์ฆ๊ฐ ๋ฐํ๋๋ฉฐ ์ด์ ๋ฆด๋ฆฌ์ฆ์์ ํธํ์ฑ์ ์ ์งํ๊ณ ์์ต๋๋ค.
-
๋ฆด๋ฆฌ์ฆ๋ 3๊ฐ์ง ๋ฒ์ ์ผ๋ก ์ ๊ณต๋ฉ๋๋ค: โstableโ, โbetaโ ๊ทธ๋ฆฌ๊ณ โnightlyโ.
-
์๋ก์ด ๊ธฐ๋ฅ์ โnightlyโ -> โbetaโ -(6์ฃผ ํ)-> โstableโ ๋ก ๋ณ๊ฒฝ๋ฉ๋๋ค.
-
์์กด์ฑ์ ๋ค์ํ ์ ์ฅ์, git ํ๋ก์ ํธ, ๋๋ ํฐ๋ฆฌ ๋ฑ์์ ์ ๊ณต๋ ์ ์์ต๋๋ค.
-
๋ฌ์คํธ๋ ์๋์ ์ผ๋ก ๊ตฌ๋ถ๋ฉ๋๋ค. ํ์ฌ๋ Rust 2021 ์๋์ ์ ๋๋ค. ์ด ์ ์๋์ ์ผ๋ก Rust 2015์ Rust 2018์ด ์์ต๋๋ค.
-
์๋์ ์ ์ด์ ์๋์ ๊ณผ ํธํ์ด ๋์ง ์์ ์ ์์ต๋๋ค.
-
์๋์ ์ด ๋ฐ๋๋ฉด์ ํ๋ก๊ทธ๋จ์ด ์๋์น ์๊ฒ ๊นจ์ง๋ ๋ฌธ์ ๋ฅผ ๋ง๊ธฐ ์ํด, ๊ฐ ํ๋ก๊ทธ๋จ์ ์์ ์ด ๋น๋๋ ์๋์ ์ ๋ช ์์ ์ผ๋ก
Cargo.toml
์ ์ง์ ํด์ผ ํฉ๋๋ค. -
๋ฌ์คํธ ์ํ๊ณ๊ฐ ์๋์ ๋ณ๋ก ํํธํ ๋๋ ๊ฒ์ ๋ง๊ธฐ ์ํด, ๋ฌ์คํธ ์ปดํ์ผ๋ฌ๋ ์๋ก ๋ค๋ฅธ ์๋์ ์์ ์์ฑ๋ ์ฝ๋๋ค์ ํ๋์ ๋ฐ์ด๋๋ฆฌ๋ก ๋ฌถ์ ์ ์์ต๋๋ค.
-
cargo
๋ฅผ ์ฌ์ฉํ์ง ์๊ณ ์ปดํ์ผ๋ฌ๋ฅผ ์ง์ ์ฌ์ฉํ๋ ๊ฒฝ์ฐ๋ ๊ฑฐ์ ์์์ ์ธ๊ธํด ์ฃผ์๊ธฐ ๋ฐ๋๋๋ค. -
์นด๊ณ ์์ฒด๊ฐ ๋งค์ฐ ๊ฐ๋ ฅํ๊ณ ํฌ๊ด์ ์ธ ๋๊ตฌ์์ ์ ๊ทน์ ์ผ๋ก ์๋ฆฌ์ธ์. ์นด๊ณ ๋ ๋ค์๊ณผ ๊ฐ์ ๋ค์ํ ๊ณ ๊ธ ๊ธฐ๋ฅ์ ์ ๊ณตํฉ๋๋ค:
- ํ๋ก์ ํธ/ํจํค์ง ๊ตฌ์กฐํ
- ์ํฌ์คํ์ด์ค
- ๊ฐ๋ฐ/๋ฐํ์ ์ข ์์ฑ ๊ด๋ฆฌ ๋ฐ ์บ์ฑ
- ๋น๋ ์คํฌ๋ฆฝํธ
- ์ ์ญ ์ค์น
- cargo clippy์ ๊ฐ์ ํ์ ํ๋ฌ๊ทธ์ธ์ผ๋ก ํ์ฅ ๊ฐ๋ฅ.
-
๊ณต์ Cargo Book์์ ์์ธํ ์ฌํญ์ ํ์ธํ์๊ธฐ ๋ฐ๋๋๋ค.
-
๊ฐ์์์์ ์ฝ๋ ์ํ
์ด ๊ฐ์์๋ฃ์ ์๋ ๋ชจ๋ ์์ ๋ ๋ธ๋ผ์ฐ์ ์์ ๋ฐ๋ก ์ํ ๊ฐ๋ฅํฉ๋๋ค. ์ด๋ ๊ฒ ํ ์ด์ ๋, ์ค๋น ๊ณผ์ ์ ๋จ์ํ ์ํค๊ณ , ๋ชจ๋๊ฐ ๊ฐ์ ํ๊ฒฝ์์ ์์ ํ ์ ์๋๋ก ํ๊ธฐ ์ํจ์ ๋๋ค.
๊ทธ๋ผ์๋ ๋ถ๊ตฌํ๊ณ , ์นด๊ณ (cargo)๋ฅผ ์ง์ ์ค์นํ๋ ๊ฒ์ ๊ฐ๋ ฅ ๊ถ์ฅํฉ๋๋ค. ์ด๊ฒ ๊ณผ์ ์์ฑ์ ๋ ๋์์ด ๋ ๊ฒ๋๋ค. ๋ํ, ๋ง์ง๋ง ๋ ์๋ ์์กด์ฑ์ด ์๋ ์์ ๋ฅผ ์์ ํ๊ฒ ๋ ํ ๋ฐ, ๊ทธ ๋์๋ ์ด์ฐจํผ ์นด๊ณ ๊ฐ ํ์ํฉ๋๋ค.
์ด ๊ฐ์ ์๋ฃ์ ์ฝ๋ ๋ธ๋ก๋ค์ ์ ๋ถ ์ธํฐ์ํฐ๋ธ ํฉ๋๋ค:
fn main() { println!("Edit me!"); }
์ฝ๋ ๋ธ๋ก์ ํฌ์ปค์ค๋ฅผ ๋๊ณ Ctrl + Enter๋ฅผ ๋๋ฌ ์คํํด ๋ณผ ์ ์์ต๋๋ค.
๊ฐ์์์ ๋๋ถ๋ถ์ ์ฝ๋ ์ํ์ ์์ ๊ฐ์ด ์์ ํ ์ ์์ง๋ง ์ผ๋ถ ์ฝ๋๋ ๋ค์๊ณผ ๊ฐ์ ์ด์ ๋ก ์์ ํ ์ ์์ต๋๋ค:
-
์ ๋ ํ ์คํธ๋ ๋ด์ฅ ํ๋ ์ด๊ทธ๋ผ์ด๋์์ ์คํ์ด ์๋ฉ๋๋ค. ์ธ๋ถ ํ๋ ์ด๊ทธ๋ผ์ด๋ ์ฌ์ดํธ์ ๋ถ์ฌ๋ฃ์ด ํ ์คํธ๋ฅผ ์คํํ์๊ธฐ ๋ฐ๋๋๋ค.
-
๋ด์ฅ๋ ํ๋ ์ด๊ทธ๋ผ์ด๋์์๋ ํ์ด์ง ์ด๋์ ์์ฑ๋ ๋ชจ๋ ๋ด์ฉ์ด ์ฌ๋ผ์ง๋๋ค. ๋ฐ๋ผ์ ๋ก์ปฌ ํ๊ฒฝ์ด๋ ์ธ๋ถ ํ๋ ์ด๊ทธ๋ผ์ด๋ ์ฌ์ดํธ์์ ์ฐ์ต๋ฌธ์ ๋ฅผ ํด๊ฒฐํ๋ ๊ฒ์ด ์ข์ต๋๋ค.
๋ก์ปฌ ํ๊ฒฝ์ ์นด๊ณ
๋ง์ฝ ๊ฐ์ธ์ฉ ์ปดํจํฐ์์ ์ฝ๋๋ฅผ ์คํํด๋ณด๋ ค๋ฉด ๋จผ์ ๋ฌ์คํธ๋ฅผ ์ค์นํด์ผ ํฉ๋๋ค. Rust Book์ ์ง์นจ์ ๋ฐ๋ผ rustc
์ cargo
๋ฅผ ํจ๊ป ์ค์น ํ์๊ธฐ ๋ฐ๋๋๋ค. ์ค์น ํ ์๋ ์ปค๋งจ๋๋ฅผ ํตํด ๊ฐ ํด์ ๋ฒ์ ์ ํ์ธ ํ ์ ์์ต๋๋ค:
% rustc --version
rustc 1.69.0 (84c898d65 2023-04-16)
% cargo --version
cargo 1.69.0 (6e9a83356 2023-04-12)
์ด ๋ฒ์ ๋ณด๋ค ๋ ์ต์ ์ ๋ฒ์ ์ด์ด๋ ์๊ด ์์ต๋๋ค. ๋ฌ์คํธ๋ ํ์ ํธํ์ฑ์ ์ง์ํฉ๋๋ค.
์ ์์ ์ผ๋ก ์ค์น๊ฐ ๋์์ผ๋ฉด, ๊ฐ์ ์์ ์ค ํ๋๋ฅผ ๋ฌ์คํธ ๋ฐ์ด๋๋ฆฌ๋ก ๋น๋ํด ๋ด ์๋ค:
-
์์ ๋ธ๋ก์ ์๋ โCopy to clipboardโ ๋ฒํผ์ ํด๋ฆญํด์ ๋ณต์ฌํฉ๋๋ค.
-
ํฐ๋ฏธ๋์์
cargo new exercise
๋ฅผ ์ ๋ ฅํด์ ์๋ก์ดexercise/
ํด๋๋ฅผ ๋ง๋ญ๋๋ค:$ cargo new exercise Created binary (application) `exercise` package
-
exercise/
ํด๋๋ก ์ด๋ํ ํ,cargo run
์ปค๋งจ๋๋ก ์ฝ๋๋ฅผ ์คํํฉ๋๋ค:$ cd exercise $ cargo run Compiling exercise v0.1.0 (/home/mgeisler/tmp/exercise) Finished dev [unoptimized + debuginfo] target(s) in 0.75s Running `target/debug/exercise` Hello, world!
-
src/main.rs
์ ์ฝ๋๋ฅผ ์์ฑํฉ๋๋ค. ์๋ฅผ ๋ค์ด ์ด์ ํ์ด์ง์ ์์ค๋ฅผ ์๋์ ๊ฐ์ดsrc/main.rs
์ ์์ฑํฉ๋๋คfn main() { println!("Edit me!"); }
-
cargo run
์ปค๋งจ๋๋ก ์์ค๋ฅผ ๋น๋ํ๊ณ ์คํํฉ๋๋ค:$ cargo run Compiling exercise v0.1.0 (/home/mgeisler/tmp/exercise) Finished dev [unoptimized + debuginfo] target(s) in 0.24s Running `target/debug/exercise` Edit me!
-
cargo check
์ปค๋งจ๋๋ ๋น ๋ฅด๊ฒ ์๋ฌ๋ฅผ ํ์ธํ ์ ์์ต๋๋ค.cargo build
๋ ์คํ์์ด ์ปดํ์ผ๋ง ํฉ๋๋ค. ์ด ๊ฒฝ์ฐ์target/debug/
ํด๋์์ output์ ํ์ธ ํ ์ ์์ต๋๋ค.cargo build --release
์ปค๋งจ๋๋ ๋ฆด๋ฆฌ์ฆ ๋ฒ์ ์ฉ ์ต์ ํ๋ฅผ ์ผ์ ์ปดํ์ผํ๋ฉฐtarget/release/
ํด๋์์ ํ์ธ ํ ์ ์์ต๋๋ค. -
Cargo.toml
ํ์ผ์๋ ์์กด์ฑ ํจํค์ง๋ฅผ ์ถ๊ฐํ ์ ์์ต๋๋ค.cargo
์ปค๋งจ๋๋ฅผ ์คํํ๋ฉด ์๋์ผ๋ก ์์กด์ฑ ํจํค์ง๋ฅผ ๋ค์ด๋ก๋ํ๊ณ ์ปดํ์ผ ๊น์ง ํด ์ค๋๋ค.
์๊ฐ์๋ค์ด ์นด๊ณ ๋ฅผ ์ค์นํ๊ณ ๋ก์ปฌ ํธ์ง๊ธฐ๋ฅผ ์ด์ฉํ๋๋ก ๋ ๋ คํ์ธ์. ์กฐ๊ธ ๊ท์ฐฎ์ ์๋ ์์ง๋ง, ์ด๋ ๊ฒ ํด์ผ๋ง ์ข ๋ ์ค์ ์ ๊ฐ๊น์ด ๊ฐ๋ฐํ๊ฒฝ์ ๊ฐ์ถ๊ฒ ๋๋ ๊ฒ์ ๋๋ค.
1์ผ์ฐจ ๊ฐ์
๊ฐ์ ์ฒซ ๋ ์ ๋๋ค. ์ค๋ ๋ฐฐ์ธ ๊ฒ์ด ์ฐธ ๋ง์ต๋๋ค:
-
๋ฌ์คํธ ๊ธฐ๋ณธ ๋ฌธ๋ฒ: ๋ณ์, ์ค์นผ๋ผ / ๋ณตํฉ ํ์ , ์ด๊ฑฐํ, ๊ตฌ์กฐ์ฒด, ์ฐธ์กฐํ, ํจ์์ ๋ฉ์๋.
-
ํ๋ฆ ์ ์ด:
if
,if let
,while
,while let
,break
, ๊ทธ๋ฆฌ๊ณcontinue
. -
ํจํด ๋งค์นญ: ์ด๊ฑฐํ, ๊ตฌ์กฐ์ฒด ๊ทธ๋ฆฌ๊ณ ๋ฐฐ์ด ๋ถํด.
ํ์๋ค์๊ฒ ๋ค์์ ์๊ธฐ์์ผ ์ฃผ์๊ธฐ ๋ฐ๋๋๋ค:
- ๊ถ๊ธํ ์ ์ด ์์ผ๋ฉด ์ฃผ์ ํ์ง ๋ง๊ณ ์ง๋ฌธ ํด์ผ ํฉ๋๋ค.
- ์ด ์์
์ ์ํธ์์ฉ์ด ์ํฉ๋๋ค. ํ ๋ก ์ ๋ง์ค์ด์ง ๋ง์ธ์!
- ๊ฐ์ฌ๋ก์ ํ ๋ก ์ด ์๊ธธ๋ก ์์ง ์๊ฒ ์ฃผ์ํ์ธ์. ์๋ฅผ ๋ค์ด ๋ฌ์คํธ์ ๋ค๋ฅธ ์ธ์ด๋ค์ ๋น๊ตํ๋ค๋ ์ง ํ๋ ๊ฒ์ ์ข์ต๋๋ค. ์ ์ ํ ๊ท ํ์ ์ฐพ๊ธฐ ์ ๋งคํ ๊ฒฝ์ฐ๋ผ๋ฉด ํ ๋ก ์ ํ์ฉํ๋ ์ชฝ์ด ์ผ๋ฐฉ์ ์ธ ๊ฐ์๋ณด๋ค๋ ๋ ๋ง์ ์ฌ๋๋ค์ ์ฐธ์ฌ๋ฅผ ์ด๋์ด ๋ผ ์ ์์ต๋๋ค.
- ์ง๋ฌธ์ด ์ฌ๋ผ์ด๋๋ณด๋ค ์์๊ฐ๋ ๊ด์ฐฎ์ต๋๋ค.
- ํ์ต์ ์์ด์ ๋ฐ๋ณต์ ๋งค์ฐ ์ค์ํฉ๋๋ค. ์ฌ๋ผ์ด๋๋ ๊ทธ์ ๋์์ ์ค ๋ฟ, ์ํ๋ ๋๋ก ๊ฑด๋๋์ด๋ ๋ฉ๋๋ค.
์ฒซ ๋ ๊ฐ์์ ๋ชฉํ๋, ๋ฌ์คํธ์์ ๊ทธ ์ ๋ช ํ ๋น๋ฆผ ํ์ธ์ ๋ํด์ ์ด์ผ๊ธฐ ํ ์ ์์ ์ ๋ ๊น์ง๋ง ๋ฌ์คํธ๋ฅผ ์๊ฐํ๋ ๊ฒ์ ๋๋ค. ๋ฌ์คํธ์ ๊ฐ์ฅ ๋ ํนํ ํน์ง์ด ๋ฉ๋ชจ๋ฆฌ๋ฅผ ๋ค๋ฃจ๋ ๋ฐฉ์์ด๊ธฐ ๋๋ฌธ์, ํ์๋ค์๊ฒ ์ด ๋ถ๋ถ ์ ์ฐ์ ์ ์ผ๋ก ๋ณด์ฌ์ฃผ๋ ค ํฉ๋๋ค.
๋ง์ฝ ๋น์ ์ด ๊ฐ์์ค์์ ๊ฐ๋ฅด์น๊ณ ์๋ค๋ฉด, ์ด ์ฌ๋ผ์ด๋๋ ์ผ์ ์ ๊ฒํ ํ๊ธฐ์ ์ ํฉํ ๊ณณ์ ๋๋ค. ํ๋ฃจ์น ๊ฐ์๋ฅผ ์๋์ฒ๋ผ ์ค์ ์คํ๋ก ๋๋์ด ์งํํ๋ ๊ฒ์ ์ถ์ฒํฉ๋๋ค. (์ฌ๋ผ์ด๋๊ฐ ๊ทธ๋ฐ์์ผ๋ก ๋๋์ด ์์ต๋๋ค.)
- ์ค์ : 9:00 ~ 12:00,
- ์คํ: 13:00 ~ 16:00.
๋ฌผ๋ก ํ์์ ๋ฐ๋ผ ์กฐ์ ํ ์ ์์ต๋๋ค. ๊ฐ์ ์ค๊ฐ์ ์ฌ๋์๊ฐ์ ๋ฃ๋ ๊ฒ์ ์์ง ๋ง์ธ์. ๋งค ์๊ฐ ํด์์ ๊ฐ๋๊ฑธ ์ถ์ฒํฉ๋๋ค!
๋ฌ์คํธ๋?
๋ฌ์คํธ๋ 2015๋ ์ ๋ฒ์ 1.0์ ๋ฆด๋ฆฌ์ฆ ํ ์๋ก์ด ํ๋ก๊ทธ๋จ ์ธ์ด์ ๋๋ค:
- ๋ฌ์คํธ๋ C++์ ์ ์ฌํ ์ ์ ์ปดํ์ผ ์ธ์ด์
๋๋ค
rustc
๋ LLVM์ ๋ฐฑ์๋๋ก ์ฌ์ฉํฉ๋๋ค.
- ๋ฌ์คํธ๋ ๋ค์ํ ํ๋ซํผ๊ณผ ์ํคํ
์ณ๋ฅผ ์ง์ํฉ๋๋ค:
- x86, ARM, WebAssembly, โฆ
- Linux, Mac, Windows, โฆ
- ๋ฌ์คํธ๋ ๋ค์ํ ์ฅ์น์์ ์ฌ์ฉ๋ ์ ์์ต๋๋ค:
- ํ์จ์ด์ ๋ถํธ๋ก๋(์๋ฒ ๋๋)
- ์ค๋งํธ ๋์คํ๋ ์ด,
- ์ค๋งํธํฐ,
- ๋ฐ์คํฌํ,
- ์๋ฒ.
๋ฌ์คํธ๋ C++๊ฐ ์ฌ์ฉ๋๋ ๋๋ถ๋ถ์ ๊ณณ์์ ์ฌ์ฉ ๊ฐ๋ฅํฉ๋๋ค:
- ๋์ ์ ์ฐ์ฑ.
- ๋์ ์์ค์ ์ ์ด.
- ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ ๊ฐ์ ๋งค์ฐ ์ ํ๋ ์ฅ์น๋ก ์ค์ผ์ผ ๋ค์ด ๊ฐ๋ฅ.
- ๋ณ๋์ ๋ฐํ์์ ํ์๋ก ํ์ง ์์ผ๋ฉฐ, ๊ฐ๋น์ง ์ปฌ๋ ์ ๋ ์์.
- ์ฑ๋ฅ์ ํํํ์ง ์์ผ๋ฉด์๋ ์์ ์ฑ๊ณผ ์์ ์ ์ค์ ์ ๋ .
Hello World!
๊ฐ์ฅ ๊ฐ๋จํ ๋ฌ์คํธ ํ๋ก๊ทธ๋จ์ผ๋ก์จ, ๊ณ ์ ์ ์ธ Hello World ๋ฅผ ์์ฑํด ๋ณด๊ฒ ์ต๋๋ค:
fn main() { println!("Hello ๐!"); }
ํ์ธํ ์ ์๋ ๊ฒ๋ค:
- ํจ์๋
fn
์ผ๋ก ์ ์ธํฉ๋๋ค. - C/C++ ์ ๋ง์ฐฌ๊ฐ์ง๋ก ์ค๊ดํธ
{}
๋ก ๋ธ๋ก์ ํ์ํฉ๋๋ค. main
ํจ์๋ ํ๋ก๊ทธ๋จ ์ง์ ์ ์ ๋๋ค.- ๋ฌ์คํธ๋ ๋๋ํ ๋งคํฌ๋ก(hygienic macros) ์์คํ
์ ๊ฐ์ง๊ณ ์์ต๋๋ค.
println!
๋ ๊ทธ ์์์ ๋๋ค. - ๋ฌ์คํธ์ ๋ฌธ์์ด์ UTF-8๋ก ์ธ์ฝ๋ฉ๋๋ฉฐ ์ด๋ชจ์ง์ ๊ฐ์ ์ ๋์ฝ๋ ๋ฌธ์๋ฅผ ํฌํจํ ์ ์์ต๋๋ค.
์ด ์ฌ๋ผ์ด๋๋ ํ์๋ค์ด ๋ฌ์คํธ ์ฝ๋์ ์ต์ํด์ง๊ธฐ ์ํด ์์ฑ๋์์ต๋๋ค. ์์ผ๋ก 3์ผ ๋์ ๋ง์ ์ฝ๋๋ฅผ ์ ํ ๊ฒ์ด๊ธฐ ๋๋ฌธ์ ์ฐ์ ์น์ํ ์ฝ๋๋ถํฐ ์์ํฉ๋๋ค.
ํค ํฌ์ธํธ:
-
๋ฌ์คํธ๋ C/C++/Java์ ๊ฐ์ ์ ํต์ ์ธ ๋ค๋ฅธ ์ธ์ด์ ๋งค์ฐ ์ ์ฌํฉ๋๋ค. ๋ฌ์คํธ๋ ์ ์ฐจ์ ์ธ์ด์ ๋๋ค. ์ ๋ง๋ก ํ์ํ ๊ฒฝ์ฐ๊ฐ ์๋๋ผ๋ฉด, ๋ฌ์คํธ๋ ์ด๋ฏธ ์กด์ฌํ๋ ๊ฒ์ ์๋ก ๊ตฌํํ๋ ค๊ณ ํ์ง ์์ต๋๋ค.
-
๋ฌ์คํธ๋ ์ ๋์ฝ๋ ์ง์๊ณผ ๊ฐ์ ํ๋ ์ธ์ด์ ํน์ง์ ์ ๋ถ ์ง์ํฉ๋๋ค.
-
๋ฌ์คํธ๋ ์ธ์์ ๊ฐ์๋ฅผ ์ฌ์ ์ ์ง์ ํ ์ ์๋ ์ํฉ์์ ํจ์ ์ค๋ฒ๋ก๋ฉ๋์ ๋งคํฌ๋ก๋ฅผ ์ฌ์ฉํฉ๋๋ค.
-
๋๋ํ ๋งคํฌ๋ก(hygienic macro)๋ ๋งคํฌ๋ก๊ฐ ์ฌ์ฉ๋๋ ์ค์ฝํ์์ ์๋์น ์๊ฒ ๋ณ์๋ฅผ ๊ฐ๋ก์ฑ์ง ์์ต๋๋ค. ์ฌ์ค ๋ฌ์คํธ ๋งคํฌ๋ก๋ ์์ ํ hygenicํ์ง๋ ์์ต๋๋ค. ๋งํฌ๋ฅผ ์ฐธ๊ณ ํ์ธ์.
-
๋ฌ์คํธ๋ ๋ฉํฐ ํจ๋ฌ๋ค์ ์ธ์ด์ ๋๋ค. ์๋ฅผ ๋ค์ด ๊ฐ๋ ฅํ ๊ฐ์ฒด ์งํฅ ํ๋ก๊ทธ๋๋ฐ ๊ธฐ๋ฅ์ ์ง์ํ๊ธฐ๋ ํ๋ฉฐ, ํจ์ํ ์ธ์ด๋ก ๋ถ๋ฅ๋์ง๋ ์์ง๋ง ํญ๋์ ๋ฒ์์ ํจ์ํ ์ปจ์ ์ ์ง์ํฉ๋๋ค.
์์ ์์
๋ฌ์คํธ๋ก ์์ฑ๋ ์์ ์์ ์ ๋๋ค:
fn main() { // Program entry point let mut x: i32 = 6; // Mutable variable binding print!("{x}"); // Macro for printing, like printf while x != 1 { // No parenthesis around expression if x % 2 == 0 { // Math like in other languages x = x / 2; } else { x = 3 * x + 1; } print!(" -> {x}"); } println!(); }
์ด ์ฝ๋๋ ์ฝ๋ผ์ธ ์ถ์ธก(Collatz conjecture)์ผ๋ก ๊ตฌํ๋ฉ๋๋ค: ๋ฐ๋ณต๋ฌธ์ด ์ธ์ ๋ ์ข ๋ฃ๋ ๊ฒ์ด๋ผ๊ณ ๋ฏฟ์ง๋ง ์ฆ๋ช ๋ ๊ฒ์ ์๋๋๋ค. ์ฝ๋๋ฅผ ์์ ํ๊ณ ์คํํด ๋ณด์๊ธฐ ๋ฐ๋๋๋ค.
ํค ํฌ์ธํธ:
-
๋ชจ๋ ๋ณ์๊ฐ ์ปดํ์ผ ์ ์ ํด์ง ํ์ ์ ๊ฐ์ง์ ์ค๋ช ํฉ๋๋ค.
i32
๋ฅผ ์ญ์ ํ์ฌ ์ปดํ์ผ๋ฌ๊ฐ ํ์ ์ถ๋ก ์ ํ๋๋ก ํด ๋ด ๋๋ค.i32
์i8
๋ก ๋ณ๊ฒฝํ์ฌ ๋ฐํ์ ์ค๋ฒํ๋ก๋ฅผ ์ ๋ฐํด ๋ณผ ์ ์์ต๋๋ค. -
let mut x
๋ฅผlet x
๋ก ์์ ํ์ฌ ์ปดํ์ผ ์ค๋ฅ์ ๋ํด ํ ๋ก ํฉ๋๋ค. -
์ธ์๊ฐ ํฌ๋งท ๋ฌธ์์ด๊ณผ ์ผ์นํ์ง ์๋ ๊ฒฝ์ฐ
print!
์์ ์ปดํ์ผ ์ค๋ฅ๊ฐ ๋ฐ์ํจ์ ์ธ๊ธํ๋ ๊ฒ๋ ์ข์ต๋๋ค. -
๋จ์ผ ๋ณ์๋ณด๋ค ๋ณต์กํ ์์ ์ถ๋ คํ๋ ค๋ฉด
{}
์ ์๋ฆฌ ํ์์๋ก ์ฌ์ฉํ๋ ๋ฐฉ๋ฒ์ ๋ณด์ฌ ์ค๋๋ค. -
ํ์๋ค์๊ฒ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ๊ฐ ์ด๋ ์๋์ง ์๋ ค ์ฃผ๊ณ ๋,
print!
๊ฐ ์ง์ํ๋ ํฌ๋งทํ ์ธ์ด์ ๋ฌธ๋ฒ์ ์๊ธฐ ์ํดstd::fmt
๋ฅผ ๊ฒ์ํด์ผ ํ๋ค๋ ๊ฒ์ ๊ฐ๋ฅด์น์ธ์.ํ์๋ค์ด ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ๊ฒ์ ๊ธฐ๋ฅ์ ์ต์ํด ์ง๋๋ก ํ๋ ๊ฒ์ด ์ค์ํฉ๋๋ค.- ์์์
rustup doc std::fmt
๋ฅผ ์ํํ๋ฉด ๋ก์ปฌ ๋ธ๋ผ์ฐ์ ๋กstd:fmt
์ ๋ํ ๋ฌธ์๋ฅผ ๋ณด์ฌ์ค๋๋ค
- ์์์
๋ฌ์คํธ๋ฅผ ์จ์ผํ๋ ์ด์
๋ฌ์คํธ๋ง์ ๋ ํนํ ์ธ์ผ์ฆ ํฌ์ธํธ(์ฅ์ ):
- ์ปดํ์ผ ์ ๋ฉ๋ชจ๋ฆฌ ์์ ์ด ๋ณด์ฅ๋จ.
- ์ ์๋์ง ์์ ๋ฐํ์ ๋์์ด ์์.
- ํ๋์ ์ธ ์ธ์ด ๊ธฐ๋ฅ.
์๊ฐ์๋ค์๊ฒ ์ด๋ค ํ๋ก๊ทธ๋๋ฐ ์ธ์ด๋ฅผ ์ฌ์ฉํ๋์ง ๋ฌผ์ด๋ณด์๊ธฐ ๋ฐ๋๋๋ค. ์ด๋ค ์ธ์ด๋ฅผ ์ฌ์ฉํ๋๋์ ๋ฐ๋ผ ๋ฌ์คํธ์์ ์ด๋ค ์ ์ ๊ฐ์กฐํด์ผ ํ ์ง๋ฅผ ๊ณ ๋ฏผํด ๋ณด์ธ์:
-
C/C++: ๋ฌ์คํธ๋ โ๋น๋ฆผโ๊ฒ์ฌ๊ธฐ๋ฅผ ํตํด์ ์ํ์ค์ ๋ฐ์ํ ์ ์๋ ๋ชจ๋ ์๋ฌ๋ฅผ ์ ๊ฑฐํฉ๋๋ค. ๋ฌ์คํธ๋ C์ C++๊ณผ ๋น์ทํ ์์ค์ ์ฑ๋ฅ์ ๋ณด์ฌ์ฃผ๋ฉด์๋, ๊ทธ ์ธ์ด๋ค์์ ์ข ์ข ๋ฐ์ํ๋ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ จ ์ค๋ฅ๊ฐ ์์ต๋๋ค. ๋ํ, ํจํด ๋งค์นญ์ด๋, ๊ธฐ๋ณธ์ ์ผ๋ก ์ ๊ณต๋๋ ์ข ์์ฑ ๊ด๋ฆฌ์ ๊ฐ์ ํ๋์ ์ธ ์ธ์ด์ ๊ธฐ๋ฅ๋ค์ ์ ๊ณตํฉ๋๋ค.
-
Java, Go, Python, JavaScript: ์ด ์ธ์ด๋ค๊ณผ ๋์ผํ ๋ฉ๋ชจ๋ฆฌ ์์ ์ฑ๊ณผ ํจ๊ป, โํ์ด๋ ๋ฒจโ์ธ์ด์ ๋๋์ ๋๋ ์ ์์ต๋๋ค. ๊ฑฐ๊ธฐ์ ๋ํด, ๊ฐ๋น์ง ์ปฌ๋ ํฐ๊ฐ ์๋ C/C++์ ์ ์ฌํ ์์ค์ ๋น ๋ฅด๊ณ ์์ธก ๊ฐ๋ฅํ ์ฑ๋ฅ์ ๊ธฐ๋ํ ์ ์์ต๋๋ค. ๊ทธ๋ฆฌ๊ณ ํ์ํ ๊ฒฝ์ฐ ์ ์์ค ํ๋์จ์ด๋ฅผ ๋ค๋ฃจ๋ ์ฝ๋๋ก ์์ฑํ ์ ์์ต๋๋ค.
์ปดํ์ผ ์ ๋ณด์ฅ๋๋ ๊ฒ๋ค
์ปดํ์ผ ์ ์ ์ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ:
- ์ด๊ธฐํ๋์ง ์๋ ๋ณ์๊ฐ ์์ต๋๋ค.
- ๋ฉ๋ชจ๋ฆฌ ๋์ถ ์์(๊ฑฐ์. ๊ฐ์์ฐธ์กฐ๋ ธํธ ์ฐธ๊ณ .)
- ๋ฉ๋ชจ๋ฆฌ ์ด์ค ํด์ ๊ฐ ์์ฒ์ ์ผ๋ก ๋ถ๊ฐ๋ฅ ํฉ๋๋ค.
- ๋ฉ๋ชจ๋ฆฌ ํด์ ํ ์ฌ์ฉ์ด ์์ฒ์ ์ผ๋ก ๋ถ๊ฐ๋ฅ ํฉ๋๋ค.
NULL
ํฌ์ธํฐ๋ ์์ต๋๋ค.- ๋ฎคํ ์ค๋ฅผ ์ ๊ถ ๋๊ณ ์ฌ๋ ๊ฒ์ ์๋ ์ค์๋ฅผ ํ ์ ์์ต๋๋ค.
- ์ค๋ ๋๊ฐ ๋ฐ์ดํฐ ๋ ์ด์ค๋ฅผ ๋ง์์ค๋๋ค.
- ๋ฐ๋ณต์๊ฐ ๊ฐ์๊ธฐ ๋ฌดํจํ ๋๋ ๊ฒฝ์ฐ๊ฐ ์์ต๋๋ค.
(์์ ํ) ๋ฌ์คํธ์์๋ ๋ฉ๋ชจ๋ฆฌ ๋์ถ์ด ๋ฐ์ํ ์ ์๋ ๋ช ๊ฐ์ง ๊ฒฝ์ฐ๊ฐ ์์ต๋๋ค:
Box::leak
์ ์ด์ฉํ์ฌ ํฌ์ธํฐ๋ฅผ ์๋์ ์ผ๋ก ๋์ถ์ํฌ ์ ์์ต๋๋ค. ์ด๋ฅผ ์ด์ฉํด์ ๋ฐํ์์ด ์์ฑํ๊ณ ๋ฐํ์์ด ํฌ๊ธฐ๋ฅผ ์ ํ ์ ์ ๋ณ์๋ฅผ ๊ฐ์ ธ์ฌ ์ ์์ต๋๋คstd::mem::forget
์ ์ฌ์ฉํ์ฌ ์ปดํ์ผ๋ฌ๊ฐ ๊ฐ์ ๋ํด โ์๋๋กโ ๋ง๋ค ์ ์์ต๋๋ค(์๋ฉธ์๊ฐ ์คํ๋์ง ์์์ ์๋ฏธํฉ๋๋ค).Rc
๋๋Arc
๋ฅผ ์ฌ์ฉํ์ฌ ์ค์๋ก ์ํ์ฐธ์กฐ๋ฅผ ์์ฑํ ์๋ ์์ต๋๋ค.- ์ปฌ๋ ์ ์ ๋ฌดํ์ ์ฑ์ฐ๋ ๊ฒ์ ๋ฉ๋ชจ๋ฆฌ ๋์ถ๋ก ๊ฐ์ฃผํ ์๋ ์์ง๋ง, ๋ฌ์คํธ๋ ์ด๋ฅผ ๋ณดํธํ์ง ๋ชปํฉ๋๋ค.
๋ณธ ๊ฐ์์์๋ โ๋ฉ๋ชจ๋ฆฌ ๋์ถ ์์โ์ โ์ฐ๋ฐ์ ์ธ ๋ฉ๋ชจ๋ฆฌ ๋์ถ ์์โ์ผ๋ก ์ดํดํด์ผ ํฉ๋๋ค.
๋ฐํ์ ์ ๋ณด์ฅ๋๋ ๊ฒ๋ค
๋ฐํ์ ์ ์ ์๋์ง ์์(undefined) ๋์ ์์:
- ๋ฐฐ์ด ์ ๊ทผ์ ๊ฒฝ๊ณ ์ฒดํฌ.
- ์ ์ํ ํ์ ์ ๋ณ์์์ ์ค๋ฒํ๋ก์ฐ ๋ฐ์์ ๋์์ด ์ ์ ์๋์ด์์ต๋๋ค.
ํค ํฌ์ธํธ:
-
์ ์ํ ์ค๋ฒํ๋ก์ฐ๋
overflow-checks
์ปดํ์ผ ํ์ ํ๋๊ทธ๋ฅผ ํตํด ์ ์๋ฉ๋๋ค. ์ด ํ๋๊ทธ๊ฐ ์ผ์ง๋ฉด, ํ๋ก๊ทธ๋จ์ ์ ์ํ ์ค๋ฒํ๋ก์ฐ ๋ฐ์์ panic (ํ๋ก๊ทธ๋จ์ ํฌ๋์ ์ํค๋ ์ ์ ์๋ ๋ฐฉ๋ฒ) ํฉ๋๋ค. ์ด ํ๋๊ทธ๊ฐ ๊บผ์ง๋ฉด, ์ค๋ฒํ๋ก์ฐ๋ wrap-around ๊ฐ ๋ฉ๋๋ค. ๊ธฐ๋ณธ์ ์ผ๋ก ๋๋ฒ๊ทธ ๋ชจ๋(cargo build
)์์๋ ํจ๋์ด, ๋ฆด๋ฆฌ์ฆ ๋ชจ๋(cargo build --release
)์์๋ wrap-around๊ฐ ๋ฐ์ํฉ๋๋ค. -
์ปดํ์ผ ํ๋๊ทธ๋ฅผ ์ฌ์ฉํ์ฌ ๊ฒฝ๊ณ์ฒดํฌ๋ฅผ ๋ฌด๋ ฅํ ํ ์ ์์ต๋๋ค.
unsafe
๋ฅผ ์ฌ์ฉํ๋๋ผ๋ ๋ง์ฐฌ๊ฐ์ง์ ๋๋ค. ํ์ง๋งunsafe
์์ ํธ์ถ ๊ฐ๋ฅํslice::get_unchecked
๊ฐ์ ํจ์๋ ๊ฒฝ๊ณ ๊ฒ์ฌ๋ฅผ ์ํํ์ง ์์ต๋๋ค.
ํ๋์ ์ธ ํน์ง
๋ฌ์คํธ๋ ์ง๋ ์์ญ๋ ๊ฐ์ ๋ชจ๋ (ํ๋ก๊ทธ๋๋ฐ ์ธ์ด๋ค์) ๊ฒฝํ์ผ๋ก ๋ง๋ค์ด์ก์ต๋๋ค.
์ธ์ด์ ํน์ง
- ์ด๊ฑฐํ๊ณผ ํจํด ๋งค์นญ.
- ์ ๋ค๋ฆญ.
- FFI ๋ฐํ์ ์ค๋ฒํค๋ ์์.
- ๋น์ฉ์ด ๋ค์ง ์๋ ์ถ์ํ.
๋๊ตฌ๋ค
- ์น์ ํ ์ปดํ์ผ๋ฌ ์ค๋ฅ๋ฉ์์ง.
- ๋ด์ฅ ์ข ์์ฑ ๊ด๋ฆฌ์.
- ๋ด์ฅ ํ ์คํธ ์ง์.
- LSP (Language Server Protocol, ์ธ์ด ์๋ฒ ํ๋กํ ์ฝ) ์ง์์ด ์๋์ด ์์.
ํค ํฌ์ธํธ:
-
C++ ์ ์ ์ฌํ๊ฒ ์ ๋ก ์ฝ์คํธ ์ถ์ํ๋ CPU๋ ๋ฉ๋ชจ๋ฆฌ๋ฅผ ์ฌ์ฉํ์ฌ ์์๋ ๋ฒจ ํ๋ก๊ทธ๋๋ฐ ๊ตฌ์กฐ๋ฅผ ๋ง๋๋๋ฐ โ๋น์ฉโ์ ์ง๋ถํ ํ์๊ฐ ์์ต๋๋ค. ์๋ฅผ ๋ค์ด
for
๋ฃจํ์์iter().fold()
๊ตฌ์กฐ๋ฅผ ์ฌ์ฉํ๋ ๊ฒ๊ณผ ๊ฑฐ์ ๋์ผํ ๋ฎ์ ์์ค์ ๋ช ๋ น์ด๊ฐ ์์ฑ๋ ๊ฒ ์ ๋๋ค. -
๋ฌ์คํธ์ ์ด๊ฑฐํ(enum)์ ํฉ๊ณ ํ์ (sum type)์ผ๋ก ์๋ ค์ง ๋์ํ์ ๋ฐ์ดํฐํ(Algebraic Data Type)์ผ๋ก, ํ์ ์์คํ ์ด
Option<T>
์Result<T, E>
๋ฑ์ ํํํ ์ ์๊ฒ ํด์ค๋๋ค. -
์ค๋ฅ๋ฅผ ์ฝ์ด๋ณด์๊ธฐ ๋ฐ๋๋๋ค โ ์ค๋๊ธฐ๊ฐ ๋ง์ ๊ฐ๋ฐ์๋ค์ด ์ปดํ์ผ๋ฌ ์ถ๋ ฅ์ ๋ฌด์ํ๋๋ฐ ์ต์ํด์ ธ ์์ต๋๋ค. ๋ฌ์คํธ ์ปดํ์ผ๋ฌ๋ ๋ค๋ฅธ ์ปดํ์ผ๋ฌ๋ณด๋ค ๋ ์๋ค์ค๋ฝ๊ณ , ๋ณต์ฌ-๋ถ์ฌ๋ฃ๊ธฐ ํ ์ ์๋ ์ ๋์ ์ฝ๋ ํผ๋๋ฐฑ์ ์ ๊ณตํ๋ ๊ฒฝ์ฐ๊ฐ ๋ง์ต๋๋ค.
-
๋ฌ์คํธ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ Java, Python์ด๋ Go์ ๊ฐ์ ์ธ์ด์ ๋นํด์ ๊ท๋ชจ๊ฐ ์์ต๋๋ค. ๋น์ฐํ ํฌํจ๋์ด์ผ ํ๋ค๊ณ ์๊ฐํ ์๋ ์๋ ์๋์ ๊ฐ์ ๊ฒ๋ค์ด ๋ฌ์คํธ์ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ์์ต๋๋ค:
- ๋์ ์์ฑ๊ธฐ, ํ์ง๋ง rand๋ฌธ์๋ฅผ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค.
- SSL ๋๋ TLS์ง์, ํ์ง๋ง rusttls๋ฌธ์๋ฅผ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค.
- JSON ์ง์, ํ์ง๋ง serde_json ๋ฌธ์๋ฅผ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค. ๊ทธ ์ด์ ๋ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์์ ํ ๋ฒ ์ด๋ค ๊ธฐ๋ฅ์ ์ ๊ณตํ๋ฉด ๋บ ์ ์์ผ๋ฉฐ, ๋งค์ฐ ์์ ์ ์ด์ด์ผ ํ๊ธฐ ๋๋ฌธ์ ๋๋ค. ์์ ์ธ๊ธํ ๊ธฐ๋ฅ๋ค์ ์์ง ๋ฌ์คํธ ์ปค๋ฎค๋ํฐ๊ฐ ์ต๊ณ ์ ์๋ฃจ์ ์ ์ฐพ์ง ๋ชปํ๊ธฐ ๋๋ฌธ์ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ํฌํจ๋์ง ์์์ต๋๋ค. ์ด์ฉ๋ฉด ์ด๋ค ์ค ๋ช ๊ฐ๋ โ์ต๊ณ ์ ์๋ฃจ์ โ์ด ์์ ์กด์ฌํ ์ ์์ ์ง๋ ๋ชจ๋ฆ ๋๋ค. ๋ฌ์คํธ๋ ์นด๊ณ ๋ผ๋ ํจํค์ง ๊ด๋ฆฌ์๊ฐ ๋ด์ฅ๋์ด ์๊ณ , ์๋ํํฐ ํฌ๋ ์ดํธ๋ฅผ ๋ค์ด๋ก๋, ์ปดํ์ผ ํ๊ธฐ ๋งค์ฐ ์ฝ์ต๋๋ค. ์ด ๋ํ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ๊ฐ ์์ ์ด์ ์ ๋๋ค.
์ข์ ์๋ํํฐ ํฌ๋ ์ดํธ๋ฅผ ์ฐพ๋ ๊ฒ์ ์ด๋ ต์ต๋๋ค. https://lib.rs ์ ๊ฐ์ ์ฌ์ดํธ๊ฐ ์ ๋ขฐํ ์ ์๋ ์ข์ ํฌ๋ ์ดํธ๋ฅผ ๋น๊ตํ์ฌ ์ฐพ๋๋ฐ ์ข์ต๋๋ค.
-
rust-analyzer๋ ์ฃผ์ IDE๋ ํ ์คํธ ์๋ํฐ์์ ์ฌ์ฉ๋๋ ๋ฌ์คํธ์ฉ LSP์๋ฒ ์ ๋๋ค.
๊ธฐ๋ณธ ๋ฌธ๋ฒ
๋๋ถ๋ถ์ ๋ฌ์คํธ ๋ฌธ๋ฒ์ C/C++/Java ์ ์ ์ฌํฉ๋๋ค:
- ๋ธ๋ก๊ณผ ๋ฒ์๋ ์ค๊ดํธ
{}
๋ก ํํํฉ๋๋ค. - ์ธ๋ผ์ธ ์ฃผ์์
//
, ๋ธ๋ก ์ฃผ์์/* ... */
๋ก ์ฌ์ฉํฉ๋๋ค. if
๋while
๊ฐ์ ํค์๋๋ ๋์ผํฉ๋๋ค.- ๋ณ์ ํ ๋น์
=
, ๋น๊ต๋==
๋ฅผ ์ฌ์ฉํฉ๋๋ค.
์ค์นผ๋ผ ํ์
ํ์ | ๋ฆฌํฐ๋ด ๊ฐ | |
---|---|---|
๋ถํธ์๋ ์ ์ | i8 , i16 , i32 , i64 , i128 , isize | -10 , 0 , 1_000 , 123_i64 |
๋ถํธ์๋ ์ ์ | u8 , u16 , u32 , u64 , u128 , usize | 0 , 123 , 10_u16 |
๋ถ๋์์ | f32 , f64 | 3.14 , -10.0e20 , 2_f32 |
๋ฌธ์์ด | &str | "foo" , "two\nlines" |
์ ๋์ฝ๋ ๋ฌธ์ | char | 'a' , 'ฮฑ' , 'โ' |
๋ถ๋ฆฌ์ธ | bool | true , false |
๊ฐ ํ์ ์ ํฌ๊ธฐ๋ ๋ค์๊ณผ ๊ฐ์ต๋๋ค:
iN
,uN
,fN
์ ๋ชจ๋ _N_๋นํธ ์ ๋๋ค.isize
์usize
๋ ํฌ์ธํฐ์ ๊ฐ์ ํฌ๊ธฐ์ ๋๋ค,char
32 ๋นํธ ์ ๋๋ค,bool
์ 8 ๋นํธ ์ ๋๋ค.
์์ ํ์๋์ง ์์ ๋ช ๊ฐ์ง ๋ฌธ๋ฒ์ด ์์ต๋๋ค:
-
Raw strings allow you to create a
&str
value with escapes disabled:r"\n" == "\\n"
. You can embed double-quotes by using an equal amount of#
on either side of the quotes:fn main() { println!(r#"<a href="link.html">link</a>"#); println!("<a href=\"link.html\">link</a>"); }
-
Byte strings allow you to create a
&[u8]
value directly:fn main() { println!("{:?}", b"abc"); println!("{:?}", &[97, 98, 99]); }
-
All underscores in numbers can be left out, they are for legibility only. So
1_000
can be written as1000
(or10_00
), and123_i64
can be written as123i64
.
๋ณตํฉ ํ์
ํ์ | ๋ฆฌํฐ๋ด ๊ฐ | |
---|---|---|
๋ฐฐ์ด | [T; N] | [20, 30, 40] , [0; 3] |
ํํ | () , (T,) , (T1, T2) , โฆ | () , ('x',) , ('x', 1.2) , โฆ |
๋ฐฐ์ด ์ ์ธ๊ณผ ์ ๊ทผ:
fn main() { let mut a: [i8; 10] = [42; 10]; a[5] = 0; println!("a: {:?}", a); }
ํํ ์ ์ธ๊ณผ ์ ๊ทผ:
fn main() { let t: (i8, bool) = (7, true); println!("1st index: {}", t.0); println!("2nd index: {}", t.1); }
ํค ํฌ์ธํธ:
๋ฐฐ์ด:
-
๋ฐฐ์ด์, ๊ฐ์ ํ์
T
์ ๊ฐ์ดN
๊ฐ ์๋ ๊ฒ์ ๋๋ค. ์ฌ๊ธฐ์N
์ ์ปดํ์ผ ํ์์ ๊ฒฐ์ ๋ ๊ฐ์ด์ด์ผ ํฉ๋๋ค. ์ด ๊ธธ์ด๋ ํ์ ์ ์ผ๋ถ์ ๋๋ค. ๋ฐ๋ผ์,[u8; 3]
์[u8; 4]
์ ์๋ก ๋ค๋ฅธ ํ์ ์ ๋๋ค. -
๋ฆฌํฐ๋ด์ ์ฌ์ฉํ์ฌ ๋ฐฐ์ด์ ๊ฐ์ ํ ๋นํ ์ ์์ต๋๋ค.
-
ํฌ๋งคํ ๋ฌธ์์ด์์
?
๋ ๋๋ฒ๊น ์ถ๋ ฅ์ ์๋ฏธํฉ๋๋ค.{}
๋ ๊ธฐ๋ณธ ์ถ๋ ฅ์ด๋ฉฐ,{:?}
๋ ๋๋ฒ๊น ์ถ๋ ฅ์ ๋๋ค.{a}
,{a:?}
์ ๊ฐ์ด ์ถ๋ ฅํ ๋ณ์ ์ด๋ฆ์ ํฌ๋งคํ ๋ฌธ์์ด์ ํฌํจ์ํฌ ์๋ ์์ผ๋ฉฐ, ์ด ๊ฒฝ์ฐ ์ธ์a
๋ ๋ณ๋์ ์ธ์๋ก ์ถ๊ฐํ์ง ์์ต๋๋ค. -
#
์ ์ถ๊ฐํ๋ฉด({a:#?}
) ์ข ๋ ์ฝ๊ธฐ ์ฌ์ด โ์ด์โ ํํ๋ก ์ถ๋ ฅ์ด ๋ฉ๋๋ค.
ํํ:
-
๋ฐฐ์ด๊ณผ ๋ง์ฐฌ๊ฐ์ง๋ก ํํ์ ๊ณ ์ ๊ธธ์ด๋ฅผ ๊ฐ์ต๋๋ค.
-
ํํ์ ์๋ก ๋ค๋ฅธ ํ์ ์ ๊ฐ๋ค์ ํ๋์ ๋ณตํฉ ํ์ ์ผ๋ก ๋ฌถ์ต๋๋ค.
-
ํํ์ ์ํ ๊ฐ์
t.0
,t.1
๊ณผ ๊ฐ์ด ์ธ๋ฑ์ค๋ก ์ ๊ทผํ ์ ์์ต๋๋ค. -
๋น์ด์๋ ํํ
()
์ ๋จ์ ํ์ (unit type)์ด๋ผ๊ณ ๋ ํฉ๋๋ค. ์ด๋ ํ์ ์ด๋ฉด์ ํด๋น ํ์ ์ ์ ์ผํ๋ฉฐ ์ ํจํ ๊ฐ์ ๋๋ค. ์ฆ ํ์ ๊ณผ ๊ฐ์ด ๋ชจ๋()
์ ๋๋ค. ์๋ฅผ ๋ค์ด ํจ์๋ ์์์ ๋ฐํ ๊ฐ์ด ์์์ ๋ํ๋ผ ๋ ์ฌ์ฉํฉ๋๋ค.- ๋ค๋ฅธ ์ธ์ด์์ ์ต์ํ
void
๊ฐ๋ ์ผ๋ก ์๊ฐํ ์ ์์ต๋๋ค.
- ๋ค๋ฅธ ์ธ์ด์์ ์ต์ํ
์ฐธ์กฐ
C++์ ๋ง์ฐฌ๊ฐ์ง๋ก ๋ฌ์คํธ๋ ์ฐธ์กฐํ์ ๊ฐ์ต๋๋ค:
fn main() { let mut x: i32 = 10; let ref_x: &mut i32 = &mut x; *ref_x = 20; println!("x: {x}"); }
์ฐธ๊ณ ์ฌํญ:
ref_x
์ ๊ฐ์ ํ ๋นํ ๋, C/C++์ ํฌ์ธํฐ์ ์ ์ฌํ๊ฒ*
๋ฅผ ์ด์ฉํด์ ์ฐธ์กฐ๋ฅผ ๋ฐ๋ผ๊ฐ์ผ(์ญ์ฐธ์กฐ) ํฉ๋๋ค.- ๋ฌ์คํธ๋ ํน์ ํ ๊ฒฝ์ฐ(๋ฉ์๋ ํธ์ถ)์ ์๋์ผ๋ก ์ญ์ฐธ์กฐ๋ฅผ ํฉ๋๋ค.(
ref_x.count_one()
์ ํ๋ฉด*ref_x
๊ฐcount_one
์ ์ธ์๋ก ์ ๋ฌ๋ฉ๋๋ค.) mut
๋ก ์ ์ธ๋ ์ฐธ์กฐ๋ ๊ทธ ๋ณ์๊ฐ ์ด์์๋ ๋์ ๋ค๋ฅธ ๊ฐ์ ๊ฐ์ง ์ ์์ต๋๋ค.
ํค ํฌ์ธํธ:
let mut ref_x: &i32
์let ref_x: &mut i32
์ ์ฐจ์ด์ ์ ์ฃผ์ ํ์๊ธฐ ๋ฐ๋๋๋ค. ์ฒซ๋ฒ์งธ ๊ฐ์ ๋ค๋ฅธ ๊ฐ์ ๋ฐ์ธ๋ฉ ๋ ์ ์๋ ๊ฐ๋ณ ์ฐธ์กฐ์ด๊ณ , ๋๋ฒ์งธ ๊ฐ์ ๊ฐ๋ณ ๊ฐ์ ๋ํ ์ฐธ์กฐ์ ๋๋ค.
ํ์(dangling) ์ฐธ์กฐ
๋ฌ์คํธ๋ ํ์(dangling) ์ฐธ์กฐ๋ฅผ ์ปดํ์ผ๋ฌ ๋จ๊ณ์์ ์ฐพ์๋ด๊ณ ๊ธ์งํฉ๋๋ค:
fn main() { let ref_x: &i32; { let x: i32 = 10; ref_x = &x; } println!("ref_x: {ref_x}"); }
- ์ฐธ์กฐ๋ ์ด๋ค ๊ฐ์ โ๋น๋ฆฌ๋โ ๊ฒ์ ๋๋ค.
- ๋ฌ์คํธ๋ ์ฐธ์กฐ ๋์์ ๊ฐ์ด, ๊ทธ ๊ฐ์ ๋ํ ๋ชจ๋ ์ฐธ์กฐ๋ค๋ณด๋ค ๋ ์ค๋ ์ด์์์์ ์ถ์ ํฉ๋๋ค.
- ์์ ๊ถ์ ๋ํ ์ฃผ์ ๋ฅผ ๋ค๋ฃฐ ๋ ์ด ๋น๋ฆผ์ ๋ํด ๋ ์์ธํ ์ด์ผ๊ธฐ ํ๊ฒ ์ต๋๋ค.
์ฌ๋ผ์ด์ค
์ฌ๋ผ์ด์ค๋ ํฐ ์ปฌ๋์ ์ ์ผ๋ถ(ํน์ ์ ์ฒด)๋ฅผ ๋ณด์ฌ์ฃผ๋ ๋ทฐ(view)์ ๋๋ค:
fn main() { let mut a: [i32; 6] = [10, 20, 30, 40, 50, 60]; println!("a: {a:?}"); let s: &[i32] = &a[2..4]; println!("s: {s:?}"); }
- ์ฌ๋ผ์ด์ค๋ ๋ค๋ฅธ(์ฌ๋ผ์ด์ค ๋) ํ์ ์ผ๋ก๋ถํฐ ๋ฐ์ดํฐ๋ฅผ โ๋น๋ คโ์ต๋๋ค.
- ์ง๋ฌธ:
s
๋ฅผ ์ถ๋ ฅํ๊ธฐ ์ ์a[3]
์ ์์ ํ๋ฉด ๋ฌด์จ ์ผ์ด ์์ด๋ ๊น์?
-
์ฌ๋ผ์ด์ค๋ ์ฐ์
a
๋ฅผ ๋น๋ฆฐ๋ค์, ์์๊ณผ ๋ ์ธ๋ฑ์ค๋ฅผ ๋ธ๋ํท([]
)์์ ์ง์ ํด์ ๋ง๋ญ๋๋ค. -
์ฌ๋ผ์ด์ค๊ฐ ์ธ๋ฑ์ค 0๋ถํฐ ์์ํ๋ค๋ฉด ์์ ์ธ๋ฑ์ค๋ ์๋ต ๊ฐ๋ฅํฉ๋๋ค. ์ฆ
&a[0..a.len()]
์&a[..a.len()]
๋ ๋์ผํฉ๋๋ค. -
๋ง์ง๋ง ์ธ๋ฑ์ค๋ ์๋ต ๊ฐ๋ฅํฉ๋๋ค. ๊ทธ๋์
&a[2..a.len()]
์&a[2..]
๋ ๋์ผํฉ๋๋ค. -
๋ฐ๋ผ์ ์ ์ฒด ๋ฐฐ์ด์ ๋ํ ์ฌ๋ผ์ด์ค๋
&a[..]
๊ฐ ๋ฉ๋๋ค. -
s
๋i32
๋ค๋ก ์ด๋ฃจ์ด์ง ์ฌ๋ผ์ด์ค์ ๋ํ ์ฐธ์กฐ์ ๋๋ค.s
์ ํ์ (&[i32]
)์ ๋ฐฐ์ด์ ํฌ๊ธฐ๊ฐ ๋น ์ ธ์์์ ์ฃผ๋ชฉํ์๊ธฐ ๋ฐ๋๋๋ค. ์ฆ, ์ฌ๋ผ์ด์ค๋ฅผ ์ด์ฉํ๋ฉด ๋ค์ํ ๊ธธ์ด์ ๋ฐ์ดํฐ๋ฅผ ๋ค๋ฃฐ ์ ์์ต๋๋ค. -
์ฌ๋ผ์ด์ค๋ ํญ์ ๋ค๋ฅธ ๊ฐ์ฒด๋ก๋ถํฐ โ๋น๋ คโ ์ต๋๋ค. ์ด ์์์์ ๊ฐ์ฒด
a
๋ ์ฌ๋ผ์ด์คs
๋ณด๋ค ๋ ์ค๋ ์ด์ ์์ด์ผ๋ง ํฉ๋๋ค. -
a[3]
์ ๊ฐ์ ๋ฐ๊ฟ ์ ์๋๋ ์ง๋ฌธ์ ์ข์ ์ง๋ฌธ์ ๋๋ค. ์ฌ๊ธฐ์ ๋ํ ๋ต์a
์s
๋ฅผ ํตํด ๋ฐ์ดํฐ๋ฅผ ์ฝ์ ์๋ ์์ง๋ง ์์ ํ ์๋ ์์ผ๋ฉฐ, ์ด๋ ๋ฉ๋ชจ๋ฆฌ ์์ ์ ์ํด์๋ผ๋ ๊ฒ์ ๋๋ค. ๊ทธ๋ฐ๋ฐ, ์ฌ๋ผ์ด์ค๊ฐ ์ฌ์ฉ๋์ง ์์ ๋, ์ฆ ์ฌ๋ผ์ด์ค๋ฅผ ๋ง๋ค๊ธฐ ์ ์ด๋, ํน์println
์ดํ์๋a[3]
์ ๋ฐ๊ฟ ์ ์์ต๋๋ค. ์ ๊ทธ๋ฐ์ง์ ๋ํ ์ข๋ ๊ตฌ์ฒด์ ์ธ ๋ต์ ๋น๋ฆผ ๊ฒ์ฌ ๋ถ๋ถ์์ ์์ธํ ์ค๋ช ํฉ๋๋ค.
String
๊ณผ str
์ด์ ๋ฌ์คํธ์ ๋ ๊ฐ์ง ๋ฌธ์์ด ํ์ ์ ๋ํด์ ์ดํดํด ๋ณด๊ฒ ์ต๋๋ค:
fn main() { let s1: &str = "World"; println!("s1: {s1}"); let mut s2: String = String::from("Hello "); println!("s2: {s2}"); s2.push_str(s1); println!("s2: {s2}"); let s3: &str = &s2[6..]; println!("s3: {s3}"); }
๋ฌ์คํธ ์ฉ์ด:
&str
์ ๋ฌธ์์ด ์ฌ๋ผ์ด์ค์ ๋ํ (๋ถ๋ณ) ์ฐธ์กฐ์ ๋๋ค.String
์ ๋ฌธ์์ด์ ๋ด์ ์ ์๋ ๋ฒํผ์ ๋๋ค.
-
&str
์ ๋ฌธ์์ด ์ฌ๋ผ์ด์ค์ ๋๋ค. ๋ฌธ์์ด ์ฌ๋ผ์ด์ค๋ UTF-8๋ก ์ธ์ฝ๋ฉ๋ ๋ฌธ์์ด ๋ฐ์ดํฐ๋ฅผ ์๋ฏธํฉ๋๋ค. ๋ฌธ์์ด ๋ฆฌํฐ๋ด("Hello"
)์ ํ๋ก๊ทธ๋จ ๋ฐ์ด๋๋ฆฌ์ ์ ์ฅ๋ฉ๋๋ค. -
๋ฌ์คํธ์
String
ํ์ ์ ์ค์ ๋ก๋ ๋ฌธ์์ด์ ์ด๋ฃจ๋ ๋ฐ์ดํธ์ ๋ํ ๋ฐฑํฐ(Vec<u8>
)์ ๋๋ค.Vec<T>
๊ฐT
๋ฅผ ์์ ํ๊ณ ์๋ฏ์ด,String
์ด ๊ฐ๋ฆฌํค๊ณ ์๋ ๋ฌธ์์ด์String
์ ์์ ์ ๋๋ค. -
๋ค๋ฅธ ๋ง์ ํ์ ๋ค์ฒ๋ผ
String::from
๋ ๋ฌธ์์ด ๋ฆฌํฐ๋ด๋ก๋ถํฐ ๋ฌธ์์ด์ ์์ฑํฉ๋๋ค.String::new()
๋ ์๋ก์ด ๋น ๋ฌธ์์ด์ ์์ฑํฉ๋๋ค.push()
์push_str()
๋ฉ์๋๋ฅผ ์ฌ์ฉํ์ฌ ๋ฌธ์์ด ๋ฐ์ดํฐ๋ฅผ ์ถ๊ฐ ํ ์ ์์ต๋๋ค. -
format!()
๋งคํฌ๋ก๋ ๋ณ์์ ๊ฐ์ ๋ฌธ์์ด๋ก ๋ณํํ๋ ํธ๋ฆฌํ ๋ฐฉ๋ฒ์ ๋๋ค. ์ด ๋งคํฌ๋ก๋println!()
๋งคํฌ๋ก์ ๋์ผํ ํฌ๋งทํ ํ์์ ์ง์ํฉ๋๋ค. -
&
์ ๋ฒ์ ์ฐ์ฐ์๋ฅผ ์ด์ฉํ์ฌString
์์&str
์ฌ๋ผ์ด์ค๋ฅผ ๋น๋ ค์ฌ ์ ์์ต๋๋ค. -
๋น์ ์ด C++ ํ๋ก๊ทธ๋๋จธ ๋ผ๋ฉด:
&str
๋ C++์const char*
์ ์ ์ฌํ์ง๋ง ํญ์ ์ ํจํ ๋ฌธ์์ด์ ๊ฐ๋ฆฌํจ๋ค๋ ์ ์ด ๋ค๋ฆ ๋๋ค. ๋ฌ์คํธ์String
์ C++์std::string
๊ณผ ๋๋ต ๊ฑฐ์ ๋์ผํฉ๋๋ค. (์ฃผ์ ์ฐจ์ด์ : ๋ฌ์คํธ์String
์ UTF-8 ์ธ์ฝ๋ฉ ๋ฐ์ดํธ๋ง ํฌํจํ ์ ์์ผ๋ฉฐ ์์ ๋ฌธ์์ด ์ต์ ํ(small-string optimization)๋ ๊ตฌํํ์ง ์์ต๋๋ค.
ํจ์
๋ฌ์คํธ ๋ฒ์ ์ FizzBuzz ํจ์์ ๋๋ค:
fn main() { print_fizzbuzz_to(20); } fn is_divisible(n: u32, divisor: u32) -> bool { if divisor == 0 { return false; } n % divisor == 0 } fn fizzbuzz(n: u32) -> String { let fizz = if is_divisible(n, 3) { "fizz" } else { "" }; let buzz = if is_divisible(n, 5) { "buzz" } else { "" }; if fizz.is_empty() && buzz.is_empty() { return format!("{n}"); } format!("{fizz}{buzz}") } fn print_fizzbuzz_to(n: u32) { for i in 1..=n { println!("{}", fizzbuzz(i)); } }
main
ํจ์์์ ๊ทธ ๋ค์์ ์ค๋ ํจ์๋ค์ ์ฌ์ฉํ ์ ์์ต๋๋ค. ๋ฏธ๋ฆฌ ์ ์ธํ๊ธฐ(forward declaration)๋ ํค๋ ๊ฐ์๊ฑด ํ์ ์์ต๋๋ค.- ๋งค๊ฐ๋ณ์๋ฅผ ์ ์ธํ ๋์๋ ์ด๋ฆ์ ๋จผ์ ์ฐ๊ณ , ํ์
์ ๋์ค์ ์๋๋ค. ์ด๋ฆ๊ณผ ํ์
์
:
๋ก ๊ตฌ๋ถํฉ๋๋ค. ์ด๋ ์ผ๋ถ ์ธ์ด(์๋ฅผ ๋ค์ด C)์ ๋ฐ๋์์ ์ ์ํ์๊ธฐ ๋ฐ๋๋๋ค. ๋ง์ฐฌ๊ฐ์ง๋ก, ๋ฆฌํด ํ์ ๋ ํจ์์ ์์์ด ์๋ ๊ฐ์ฅ ๋ท๋ถ๋ถ์ ์ ์ธํฉ๋๋ค. - ํจ์ ๋ณธ๋ฌธ์ ๋ง์ง๋ง ํํ์์ ๋ฐํ ๊ฐ์ด ๋ฉ๋๋ค. ๊ฐ๋จํ, ์ ๋์ ์๋
;
๋ฅผ ์๋ตํ๋ฉด ๋ฉ๋๋ค. - ๋ฐํ๊ฐ์ด ์๋ ํจ์์ ๊ฒฝ์ฐ, ์ ๋ ํ์
()
์ ๋ฐํํฉ๋๋ค.-> ()
๊ฐ ์๋ต๋ ๊ฒฝ์ฐ ์ปดํ์ผ๋ฌ๋ ์ด๋ฅผ ์ถ๋ก ํฉ๋๋ค. fizzbuzz_to()
ํจ์ ๋ดfor
๋ฐ๋ชฉ๋ฌธ์ ๋ฒ์ ํํ์ ์ค=n
์ n๊น์ง ํฌํจํ๋ค๋ ์๋ฏธ์ ๋๋ค.
Rustdoc
Rust์ ์์ดํ
(item)์ ///
๋ฌธ๋ฒ์ ์ฌ์ฉํ์ฌ ๋ฌธ์ํํ ์ ์์ต๋๋ค.
/// Determine whether the first argument is divisible by the second argument. /// /// If the second argument is zero, the result is false. fn is_divisible_by(lhs: u32, rhs: u32) -> bool { if rhs == 0 { return false; // Corner case, early return } lhs % rhs == 0 // The last expression in a block is the return value }
์ฝํ
์ธ ๋ ๋งํฌ๋ค์ด์ผ๋ก ์ฒ๋ฆฌ๋ฉ๋๋ค. ๊ฒ์๋ ๋ชจ๋ Rust ๋ผ์ด๋ธ๋ฌ๋ฆฌ ํฌ๋ ์ดํธ๋ rustdoc ๋๊ตฌ๋ฅผ ์ฌ์ฉํ์ฌ docs.rs
์ ์๋์ผ๋ก ๋ฌธ์ํ๋ฉ๋๋ค. ์ผ๋ฐ์ ์ผ๋ก API์ ๋ชจ๋ ๊ณต๊ฐ ํญ๋ชฉ์ ์ด ํจํด์ ์ฌ์ฉํ์ฌ ๋ฌธ์ํ๋ฉ๋๋ค.
-
docs.rs/rand
์์rand
ํฌ๋ ์ดํธ์ฉ์ผ๋ก ์์ฑ๋ ๋ฌธ์๋ฅผ ๋ณด์ฌ์ค๋๋ค. -
์ด ์ฌ๋ผ์ด๋์ ์์ ์ฝ๋๋ ๋ฌธ์ํ ์ฃผ์์ด ์๋ต๋์ด ์์ง๋ง, ์ค์ ์ฝ๋๋ผ๋ฉด ๋ฌธ์ํ ์ฃผ์์ ๋ฐ๋์ ์จ์ผ ํฉ๋๋ค.
-
๋ฌธ์ ๋ด๋ถ ์ฃผ์ ๋ชจ๋ ํ์ด์ง ๋ท๋ถ๋ถ์์ ๋ค๋ฃจ๋ฉฐ ์ฌ๊ธฐ์ ๋ค๋ฃจ์ง ์์๋ ๋ฉ๋๋ค.
-
๋ฌธ์ํ ์ฃผ์์ ์ฝ๋๋ฅผ ํฌํจํ ์๋ ์์ผ๋ฉฐ, ์ด ์ฝ๋๋
cargo test
๋ฅผ ํตํด ํ ์คํธ๋ก ๋์ํ ์๋ ์์ต๋๋ค. ํ ์คํธ์์ ๋ ์์ธํ ๋ค๋ฃจ๊ฒ ์ต๋๋ค.
๋ฉ์๋
๋ฉ์๋๋ ํน์ ํ์
๊ณผ ์ฐ๊ฒฐ๋ ํจ์์
๋๋ค. ๋ฉ์๋์ self
์ธ์๊ฐ ๊ทธ ๋ฉ์๋๊ฐ ์ฐ๊ฒฐ๋ ์ธ์คํด์ค์ ํ์
์
๋๋ค:
struct Rectangle { width: u32, height: u32, } impl Rectangle { fn area(&self) -> u32 { self.width * self.height } fn inc_width(&mut self, delta: u32) { self.width += delta; } } fn main() { let mut rect = Rectangle { width: 10, height: 5 }; println!("old area: {}", rect.area()); rect.inc_width(5); println!("new area: {}", rect.area()); }
- ์ค๋๊ณผ ๋ด์ผ ๊ฐ์์์ ๋ ๋ง์ ๋ฉ์๋ ์ฌ์ฉ๋ฒ์ ๋ค๋ฃฐ ๊ฒ์ ๋๋ค.
-
Rectangle::new
์์ฑ์๋ฅผ ์ถ๊ฐํ๊ณ ์ด๋ฅผmain
์์ ํธ์ถํฉ๋๋ค:fn new(width: u32, height: u32) -> Rectangle { Rectangle { width, height } }
-
_๊ธฐ์ ์ _์ผ๋ก ์ด์ผ๊ธฐ ํ์๋ฉด, ๋ฌ์คํธ๋ ์ปค์คํ ์์ฑ์๋ฅผ ์ง์ํ์ง ์์ต๋๋ค. ์ ์ ๋ฉ์๋๋ฅผ ์ฌ์ฉํ๋ ๊ฒ์ด ๊ตฌ์กฐ์ฒด๋ฅผ ์ด๊ธฐํ ํ๋ ์ผ๋ฐ์ ์ธ ๋ฐฉ๋ฒ์ ๋๋ค (๋ฌผ๋ก ์ด๊ฒ์ด ๊ฐ์ ๋์ง๋ ์์ต๋๋ค). ์ง์ง ์์ฑ์์ธ
Rectangle { width, height }
๋ฅผ ์ง์ ํธ์ถํ ์๋ ์์ต๋๋ค. ์์ธํ ๋ด์ฉ์ Rustnomicon์ ์ฐธ์กฐํ์ธ์. -
Rectangle::square(width: u32)
์์ฑ์๋ฅผ ์ถ๊ฐํ์ฌ ์์ฑ์๊ฐ ์์์ ๋งค๊ฐ๋ณ์๋ฅผ ์ฌ์ฉํ ์ ์์์ ๋ณด์ ์๋ค.
(ํจ์) ์ค๋ฒ๋ก๋ฉ
์ค๋ฒ๋ก๋ฉ์ ์ง์๋์ง ์์ต๋๋ค:
- ๊ฐ๋ณํจ์๋ ๋จ์ผ ๊ตฌํ๋ง ๊ฐ์ต๋๋ค:
- ํญ์ ๊ณ ์ ๋ ์์ ํ๋ผ๋งคํฐ๋ง ๊ฐ์ต๋๋ค.
- ํ๋ผ๋งคํฐ๋ค์ ํ์ ์ ํญ์ ๊ณ ์ ๋์ด ์์ต๋๋ค.
- ํ๋ผ๋งคํฐ์ ๊ธฐ๋ณธ ๊ฐ์ ์ง์๋์ง ์์ต๋๋ค:
- ๋ชจ๋ ํธ์ถ๋ถ์์๋ ๋์ผํ ์์ ์ธ์๋ฅผ ์ค์ ํด์ผํฉ๋๋ค.
- ์ด๋ฐ ์ฌํญ๋ค์ด ์ ์ฝ์ด ๋ ๊ฒฝ์ฐ, ๋์์ผ๋ก ๋งคํฌ๋ก๋ฅผ ์ฌ์ฉํ๊ธฐ๋ ํฉ๋๋ค.
ํ์ง๋ง, ํจ์์ ๋งค๊ฐ๋ณ์๋ ์ ๋ค๋ฆญ์ ์ ์ฉํ ์ ์์ต๋๋ค:
fn pick_one<T>(a: T, b: T) -> T { if std::process::id() % 2 == 0 { a } else { b } } fn main() { println!("coin toss: {}", pick_one("heads", "tails")); println!("cash prize: {}", pick_one(500, 1000)); }
- ์ ๋ค๋ฆญ์ ์ฌ์ฉํ ๋ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์
Into<T>
์ ํ์ ์ ๋ํ ๋คํ์ฑ์ ์ ๊ณตํ ์ ์์ต๋๋ค. ๋์ค์ ์์ธํ ์ค๋ช ํ๊ฒ ์ต๋๋ค.
1์ผ์ฐจ ์ค์ ์ฐ์ต๋ฌธ์
์ด๋ฒ ์ฐ์ต๋ฌธ์ ๋ ๋ฌ์คํธ์ ๋ ๋ถ๋ถ์ ์์๋ณผ ๊ฒ์ ๋๋ค:
-
ํ์ ์ ๋ฌต์์ ๋ณํ.
-
๋ฐฐ์ด๊ณผ
for
๋ฐ๋ณต๋ฌธ.
์ฐ์ต๋ฌธ์ ๋ฅผ ํด๊ฒฐํ๋๋ฐ ๊ณ ๋ คํด์ผ ํ ์ฌํญ๋ค:
-
๊ฐ๋ฅํ๋ค๋ฉด ๋ฌ์คํธ๊ฐ ์ค์น๋ ๋ก์ปฌ ํ๊ฒฝ์์ ์งํํ์ธ์. ๊ทธ๋ฌ๋ ํธ์ด ํ ์คํธ ์๋ํฐ์ ์๋์์ฑ ๊ธฐ๋ฅ์ ๋์์ ๋ฐ์ ์ ์์ด์ ์ข์ต๋๋ค. ์นด๊ณ ์ฌ์ฉํ๊ธฐ ์ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค.
-
ํน์ ๋ฌ์คํธ ํ๋ ์ด๊ทธ๋ผ์ด๋๋ฅผ ์ด์ฉํ ์ ์์ต๋๋ค.
ํ์ด์ง ๋ฐ์ผ๋ก ์ด๋ํ ๊ฒฝ์ฐ ์์ฑํ ๋ด์ฉ์ด ์์ค๋๊ธฐ ๋๋ฌธ์ ์ ๊ณต๋๋ ์ฝ๋ ์ค๋ํซ์ ์๋์ ์ผ๋ก ํธ์งํ ์ ์์ต๋๋ค.
After looking at the exercises, you can look at the solutions provided.
๋ฌต์์ ํ๋ณํ
๋ฌ์คํธ๋ C++ ์ ๋ค๋ฅด๊ฒ ํ์ ๊ฐ _๋ฌต์์ ๋ณํ_์ ์๋์ผ๋ก ์ ์ฉํ์ง ์์ต๋๋ค. ์๋ ์์๋ฅผ ํ์ธํด ๋ณด์ธ์:
fn multiply(x: i16, y: i16) -> i16 { x * y } fn main() { let x: i8 = 15; let y: i16 = 1000; println!("{x} * {y} = {}", multiply(x, y)); }
๋ฌ์คํธ์ ์ ์ํ ํ์
์ ๋ชจ๋ From<T>
์ Into<T>
ํธ๋ ์์ ๊ตฌํํ๊ณ ์์ผ๋ฉฐ, ์ด๋ฅผ ํตํด ํ์
๋ณํ์ด ์ด๋ฃจ์ด ์ง๋๋ค. From<T>
ํธ๋ ์์ from()
๋ฉ์๋๋ฅผ ๊ฐ์ง๊ณ ์๊ณ , Into<T>
ํธ๋ ์์ into()
๋ฉ์๋๋ฅผ ๊ฐ์ง๊ณ ์์ต๋๋ค. ๋ฌ์คํธ์์๋ From
๊ณผ Into
ํธ๋ ์์ ๊ตฌํํจ์ผ๋ก์จ, ํ์
๊ฐ ๋ณํ์ด ๊ฐ๋ฅํ๋ค๋ ๊ฒ์ ํํํฉ๋๋ค.
ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์๋ From<i8> for i16
๊ฐ ๊ตฌํ๋์ด ์๋๋ฐ ์ด๊ฒ์ i8
ํ์
์ ๋ณ์ x
๋ฅผ i16::from(x)
๋ฅผ ํธ์ถํ์ฌ i16
ํ์
์ผ๋ก ๋ณํํ ์ ์๋ค๋ ์๋ฏธ์
๋๋ค. ํน์ ๋ ๊ฐ๋จํ๊ฒ x.into()
๋ฅผ ์ฌ์ฉํ ์๋ ์์ต๋๋ค. ์ด๊ฒ์ด ๊ฐ๋ฅํ ์ด์ ๋ From<i8> for i16
๊ตฌํ์ ๊ฐ์ง๊ณ ์์ผ๋ฉด Into<i16> for i8
๊ตฌํ์ด ์๋์ผ๋ก ์์ฑ๋๊ธฐ ๋๋ฌธ์
๋๋ค.
์ด๋ ์ฌ์ฉ์ ์ ์ ํ์
์๋ ๋์ผํ๊ฒ ์ ์ฉ๋๋ ๊ท์น์
๋๋ค. ๋ฐ๋ผ์ From
๋ง์ ๊ตฌํํด๋ Into
๊น์ง ์๋์ผ๋ก ๊ตฌํ์ด ๋ฉ๋๋ค.
-
์ ์์ ์ฝ๋๋ฅผ ์คํํ๊ณ ์ด๋ค ์ปดํ์ผ ์๋ฌ๊ฐ ๋ฐ์ํ๋์ง ํ์ธํด ๋ณด์ธ์.
-
into()
๋ฅผ ์ฌ์ฉํ์ฌ ์ฝ๋๋ฅผ ์์ ํ์ธ์. -
x
์y
๋ฅผf32
์ด๋bool
,i128
๋ฑ์ผ๋ก ๋ฐ๊ฟ์ ํด๋น ํ์ ๋ค๋ก ๋ณํ์ด ๋๋์ง ํ์ธํด๋ณด์ธ์. ์์ ์ฌ์ด์ฆ ํ์ ์์ ํฐ ์ฌ์ด์ฆ๋ก ๋ณ๊ฒฝํด๋ณด์๊ณ ๊ทธ ๋ฐ๋๋ก๋ ํด๋ณด์ธ์. ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ ๋ฌธ์์์ ์๋ํด ๋ณธ ์ผ์ด์ค๊ฐ ๊ตฌํ๋์ด ์๋์ง ํ์ธํด ๋ณด์ธ์.
๋ฐฐ์ด๊ณผ for
๋ฐ๋ณต๋ฌธ
๋ฐฐ์ด์ ์๋์ ๊ฐ์ด ์ ์ธ ํ ์ ์์์ ๋ฐฐ์ ์ต๋๋ค:
#![allow(unused)] fn main() { let array = [10, 20, 30]; }
๋ฐฐ์ด์ ์ถ๋ ฅํ๋ ค๋ฉด {:?}
๋ฅผ ์๋๋ค:
fn main() { let array = [10, 20, 30]; println!("array: {array:?}"); }
๋ฌ์คํธ์์๋ for
ํค์๋๋ฅผ ์ฌ์ฉํด ๋ฐฐ์ด์ด๋ ๋ฒ์๋ฅผ ๋ฐ๋ณตํ ์ ์์ต๋๋ค:
fn main() { let array = [10, 20, 30]; print!("Iterating over array:"); for n in &array { print!(" {n}"); } println!(); print!("Iterating over range:"); for i in 0..3 { print!(" {}", array[i]); } println!(); }
์ ์ฝ๋๋ฅผ ์ด์ฉํด์, ํ๋ ฌ์ ์์๊ฒ ์ถ๋ ฅํ๋ pretty_print
ํจ์์, ํ๋ ฌ์ ์ ์น(ํ๊ณผ ์ด์ ์๋ก ๋ฐ๊พธ๋)์ํค๋ transpose
ํจ์๋ฅผ ์์ฑํด ๋ณด์๊ธฐ ๋ฐ๋๋๋ค:
๋ ํจ์ ๋ชจ๋ ํ๋ ฌ์ ํฌ๊ธฐ๋ 3 x 3 ์ผ๋ก ํ๋์ฝ๋ฉ ํฉ๋๋ค.
์๋ ์ฝ๋๋ฅผ https://play.rust-lang.org/์ ๋ณต์ฌํด์ ๊ตฌํํ์๋ฉด ๋ฉ๋๋ค:
// TODO: remove this when you're done with your implementation. #![allow(unused_variables, dead_code)] fn transpose(matrix: [[i32; 3]; 3]) -> [[i32; 3]; 3] { unimplemented!() } fn pretty_print(matrix: &[[i32; 3]; 3]) { unimplemented!() } fn main() { let matrix = [ [101, 102, 103], // <-- the comment makes rustfmt add a newline [201, 202, 203], [301, 302, 303], ]; println!("matrix:"); pretty_print(&matrix); let transposed = transpose(matrix); println!("transposed:"); pretty_print(&transposed); }
๋ณด๋์ค ๋ฌธ์
&[i32]
์ฌ๋ผ์ด์ค๋ฅผ ์ ์ด์ฉํ๋ฉด ํ๋ ฌ ํฌ๊ธฐ๋ฅผ 3 x 3์ผ๋ก ํ๋์ฝ๋ฉ ํ์ง ์์ ์ ์์๊น์? ์์ปจ๋ฐ &[&[i32]]
๋ 2์ฐจ์ ์ฌ๋ผ์ด์ค์ ์ฌ๋ผ์ด์ค ์
๋๋ค. ๊ฐ๋ฅํ๋ค๋ฉด/ํ์ง ์๋ค๋ฉด ์ ๊ทธ๋ฐ๊ฐ์?
์์ฉ ํ์ง์ ๊ตฌํ์ ๋ํด์๋ ndarray
ํฌ๋ ์ดํธ๋ฅผ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค.
๋ณด๋์ค ๋ฌธ์ ์ ๋ํ ๋ต๋ณ ์ญ์ ํด๋ต์์ ํ์ธํ ์ ์์ต๋๋ค.
The use of the reference &array
within for n in &array
is a subtle preview of issues of ownership that will come later in the afternoon.
Without the &
โฆ
- The loop would have been one that consumes the array. This is a change introduced in the 2021 Edition.
- An implicit array copy would have occurred. Since
i32
is a copy type, then[i32; 3]
is also a copy type.
ํ๋ฆ ์ ์ด
์์์ ์ดํด๋ณธ ๋ฐ์ ๊ฐ์ด ๋ฌ์คํธ์์ if
๋ ํํ์์
๋๋ค. ์กฐ๊ฑด์ ๋ฐ๋ผ ๋ ๋ธ๋ก ์ค ํ๋๋ฅผ ํ๊ฐํ๋ฉฐ, ๊ทธ ๊ฒฐ๊ณผ๊ฐ์ด if
ํํ์์ ๊ฐ์ด ๋ฉ๋๋ค. ๋ค๋ฅธ ํ๋ฆ์ ์ด ํํ์๋ ์ ์ฌํ๊ฒ ์๋ํฉ๋๋ค.
๋ธ๋ก
๋ฌ์คํธ์์ ๋ธ๋ก์ ๊ฐ๊ณผ ํ์ ์ ๊ฐ์ต๋๋ค. ๋ธ๋ก์ ๋ง์ง๋ง ํํ์์ด ๋ธ๋ก์ ๊ฐ์ด ๋ฉ๋๋ค:
fn main() { let x = { let y = 10; println!("y: {y}"); let z = { let w = { 3 + 4 }; println!("w: {w}"); y * w }; println!("z: {z}"); z - y }; println!("x: {x}"); }
์์ main
ํจ์๋ ๋ง์ง๋ง ํํ์์ด ;
๋ก ๋๋๊ธฐ ๋๋ฌธ์ ๋ฐํ๋๋ ๊ฐ๊ณผ ํ์
์ด ()
์
๋๋ค.
ํจ์์๋ ๋์ผํ ๊ท์น์ด ์ ์ฉ๋ฉ๋๋ค. ํจ์ ๋ฐ๋๋ฅผ ์ด๋ฃจ๋ ๋ธ๋ก์ ๊ฐ์ด ๋ฐํ๊ฐ์ด ๋ฉ๋๋ค:
fn double(x: i32) -> i32 { x + x } fn main() { println!("doubled: {}", double(7)); }
ํค ํฌ์ธํธ:
- ๋ฌ์คํธ์์๋ ๋ธ๋ก์ด ํ์ ๊ณผ ๊ฐ์ ๊ฐ์ง๋ค๋ ์ ์ด ์ด ์ฌ๋ผ์ด๋์ ํต์ฌ์ ๋๋ค.
- ๋ธ๋ก ๋ง์ง๋ง ์ค์ ์์ ํ๋ฉด์ ๋ธ๋ก์ ๊ฐ์ด ์ด๋ป๊ฒ ๋ฐ๋๋์ง ๋ณด์ฌ์ฃผ์ธ์. ์๋ฅผ ๋ค์ด, ์ธ๋ฏธ์ฝ๋ก ์ ๋ฃ๊ฑฐ๋ ๋บ๋ค๋ ์ง, ์๋๋ฉด
return
์ ์ฌ์ฉํด ๋ณด์ธ์.
if
ํํ์
๋ค๋ฅธ ์ธ์ด์ if
๋ฌธ๊ณผ ๋๊ฐ์ด if
ํํ์์ ์ฌ์ฉํฉ๋๋ค:
fn main() { let mut x = 10; if x % 2 == 0 { x = x / 2; } else { x = 3 * x + 1; } }
๊ฒ๋ค๊ฐ if
๋ ํํ์์ผ๋ก ์ฌ์ฉํ ์๋ ์์ต๋๋ค. ์๋ ์ฝ๋๋ ์์ ๋์ผํฉ๋๋ค:
fn main() { let mut x = 10; x = if x % 2 == 0 { x / 2 } else { 3 * x + 1 }; }
if
๋ ํํ์์ด๊ณ ํ์
์ ๊ฐ์ ธ์ผ ํ๋ฏ๋ก ๋ถ๊ธฐ ๋ธ๋ก์ ๋ชจ๋ ๊ฐ์ ํ์
์ ๊ฐ์ ธ์ผ ํฉ๋๋ค. ๋๋ฒ์งธ ์์์ x / 2
๋ค์ ;
๋ฅผ ์ถ๊ฐํ์ฌ ์ด๋ป๊ฒ ๋๋์ง ํ์ธํด ๋ณด์๊ธฐ ๋ฐ๋๋๋ค.
for
๋ฐ๋ณต๋ฌธ
for
๋ฐ๋ณต๋ฌธ์ while let
๋ฐ๋ณต๋ฌธ๊ณผ ๋งค์ฐ ์ ์ฌํฉ๋๋ค. for
๋ฐ๋ณต๋ฌธ์ ์๋์ผ๋ก into_iter()
๋ฅผ ํธ์ถํ ๋ค์ ์ด๋ฅผ ๋ฐ๋ณตํฉ๋๋ค:
fn main() { let v = vec![10, 20, 30]; for x in v { println!("x: {x}"); } for i in (0..10).step_by(2) { println!("i: {i}"); } }
๋ค๋ฅธ ์ธ์ด์ ๋ง์ฐฌ๊ฐ์ง๋ก break
์ continue
๋ฅผ ์ฌ์ฉํ ์ ์์ต๋๋ค.
- ๋ฌ์คํธ๋ ์ธ๋ฑ์ค ๊ธฐ๋ฐ์ ๋ฐ๋ณต์ ์ํ ๋ณ๋์ ๋ฌธ๋ฒ์ ์ฌ์ฉํ์ง ์์ต๋๋ค.
(0..10)
์Iterator
ํธ๋ ์์ ๊ตฌํํ๋ ๋ฒ์(range) ๊ฐ์ ๋๋ค.step_by
๋ ๋ฐ๋ณต์์ ์์๋ค์ ๊ฑด๋๋ฐ๋ ๋๋ค๋ฅธIterator
๋ฅผ ๋ฐํํ๋ ๋ฉ์๋์ ๋๋ค.- ๋ฒกํฐ ์์๋ค์ ์์ ํ๋ ค๊ณ ํ๋ฉด ๋์ค๋ ์ปดํ์ผ๋ฌ ์๋ฌ๋ฅผ ๊ฐ์ด ์ดํด๋ณด์ธ์.
v
๋ฒกํฐ๋ฅผ ๊ฐ๋ณ ๋ณ์๋ก ๋ณ๊ฒฝํ๊ณ ๋ฃจํ๋for x in v.iter_mut()
๋ก ์์ ํ์ธ์.
while
๋ฐ๋ณต๋ฌธ
while
ํค์๋๋ ๋ค๋ฅธ ์ธ์ด์ ๋งค์ฐ ๋น์ทํ๊ฒ ์๋ํฉ๋๋ค:
fn main() { let mut x = 10; while x != 1 { x = if x % 2 == 0 { x / 2 } else { 3 * x + 1 }; } println!("Final x: {x}"); }
break
์ continue
- ๋ฃจํ๋ฅผ ์กฐ๊ธฐ์ ์ข
๋ฃํ๋ ค๋ฉด
break
๋ฅผ ์ฌ์ฉํฉ๋๋ค, - ๋ค์ ๋ฐ๋ณต์ ์ฆ์ ์์ํ๋ ค๋ฉด
continue
๋ฅผ ์ฌ์ฉํฉ๋๋ค.
๋ฃจํ๋ฅผ ๋น ์ ธ๋๊ฐ๋ ค๋ฉด break
๋ฅผ, ๋ค์ ๋ฐ๋ณต์ผ๋ก ๋์ด๊ฐ๊ธฐ ์ํด์๋ continue
๋ฅผ ์ฌ์ฉํฉ๋๋ค. ์ค์ฒฉ ๋ฃจํ์์๋ ๋ ์ด๋ธ๊ณผ ํจ๊ป ์ฌ์ฉํ ์ ์์ต๋๋ค:
fn main() { let v = vec![10, 20, 30]; let mut iter = v.into_iter(); 'outer: while let Some(x) = iter.next() { println!("x: {x}"); let mut i = 0; while i < x { println!("x: {x}, i: {i}"); i += 1; if i == 3 { break 'outer; } } } }
์ ์์ ๋ ๋ด๋ถ์ while
๋ฃจํ๋ฅผ 3ํ ๋ฐ๋ณตํ ํ ๋ฐ๊นฅ ๋ฃจํ๋ฅผ ๋น ์ ธ๋๊ฐ๋๋ค.
loop
ํํ์
๋ง์ง๋ง์ผ๋ก, ๋ฌดํ ๋ฃจํ๋ฅผ ๋ง๋๋ loop
ํค์๋๊ฐ ์์ต๋๋ค.
๋ฐ๋ผ์ ๋ฐ๋์ break
๋๋ return
์ ์ฌ์ฉํด์ ๋ฃจํ๋ฅผ ์ ์งํด์ผ ํฉ๋๋ค:
fn main() { let mut x = 10; loop { x = if x % 2 == 0 { x / 2 } else { 3 * x + 1 }; if x == 1 { break; } } println!("Final x: {x}"); }
loop
๋ฅผ ๊ฐ(์:break 8
)์ผ๋ก ๋๋๊ณ ์ถ๋ ฅํฉ๋๋ค.loop
๋ non-trivial ๊ฐ์ ๋ฐํํ๋ ์ ์ผํ ๋ฐ๋ณต๋ฌธ์ ๋๋ค. ์ด๋while
๋ฐfor
๋ฐ๋ณต๋ฌธ๊ณผ ๋ฌ๋ฆฌ ์ต์ํ ํ ๋ฒ์ ๋ฃจํ๋ฌธ์ ์ํํ๋ ๊ฒ์ด ๋ณด์ฅ๋๊ธฐ ๋๋ฌธ์ ๋๋ค.
๋ณ์
๋ฌ์คํธ๋ ์ ์ ํ์ดํ์ ํตํด ํ์ ์์ ์ฑ์ ์ ๊ณตํฉ๋๋ค. ๋ณ์๋ ๊ธฐ๋ณธ์ ์ผ๋ก ๋ถ๋ณ(immutable)ํฉ๋๋ค:
fn main() { let x: i32 = 10; println!("x: {x}"); // x = 20; // println!("x: {x}"); }
- ํ์
์ถ๋ก ์ ๋๋ถ์
i32
๋ ์๋ต ๊ฐ๋ฅํฉ๋๋ค. ๊ฐ์๊ฐ ์งํ๋ ์๋ก ์๋ต ๊ฐ๋ฅํ ๋ถ๋ถ์ ์ ์ ์๋ตํ ๊ฒ์ ๋๋ค.
ํ์ ์ถ๋ก
๋ฌ์คํธ๋ ๋ณ์๊ฐ ์ด๋ป๊ฒ ์ฌ์ฉ๋๋์ง๋ฅผ ๋ณด๊ณ ๊ทธ ๋ณ์์ ํ์ ์ ์ถ๋ก ํฉ๋๋ค:
fn takes_u32(x: u32) { println!("u32: {x}"); } fn takes_i8(y: i8) { println!("i8: {y}"); } fn main() { let x = 10; let y = 20; takes_u32(x); takes_i8(y); // takes_u32(y); }
์ด ์ฌ๋ผ์ด๋๋, ๋ฌ์คํธ ์ปดํ์ผ๋ฌ๊ฐ ๋ณ์๊ฐ ์ด๋ป๊ฒ ์ ์ธ๋์ด ์๊ณ , ์ด๋ป๊ฒ ์ฌ์ฉ๋๋์ง๋ฅผ ์ ์ฝ ์กฐ๊ฑด์ผ๋ก ์ผ์์ ๋ณ์์ ํ์ ์ ์ถ๋ก ํ๋ ๋ชจ์ต์ ๋ณด์ฌ์ค๋๋ค.
์ฌ๊ธฐ์ ์ค์ํ ๊ฒ์, ์ด๋ ๊ฒ ๋ช ์์ ์ธ ํ์ ์ ์๋ตํ๊ณ ์ ์ธ๋์๋ค๊ณ ํด์ โ์ด๋ค ํ์ โ์ด๋ผ๋ ๋ค ๋ด์ ์ ์๋ ํ์ ์ด ๋๋ ๊ฒ์ ์๋๋ผ๋ ์ ์ ๋๋ค. ๋ช ์์ ์ธ ํ์ ์ ์ธ์ด ์๋ ์๋, ์ปดํ์ผ๋ฌ๊ฐ ์์ฑํ ๋จธ์ ์ฝ๋๋ ๋์ผํฉ๋๋ค. ์ปดํ์ผ๋ฌ๋ ๋จ์ง ํ์ ์ ์ธ์ ์๋ตํ ์ ์๋๋ก ํด์ ํ๋ก๊ทธ๋๋จธ๊ฐ ๋ ๊ฐ๊ฒฐํ ์ฝ๋๋ฅผ ์ธ ์ ์๋๋ก ๋์์ค ๋ฟ์ ๋๋ค.
์๋ ์ฝ๋๋, ์ ๋ค๋ฆญ ์ปจํ
์ด๋๋ฅผ ์ธ ๋ ์ปจํ
์ดํฐ ์์ ํฌํจ๋ ๋ฐ์ดํฐ์ ํ์
์ ๋ช
์์ ์ผ๋ก ์ฐ์ง ์๊ณ _
๋ก ๋์ฒดํ์ฌ๋ ๋๋ค๋ ๊ฒ์ ๋ณด์ฌ์ค๋๋ค:
fn main() { let mut v = Vec::new(); v.push((10, false)); v.push((20, true)); println!("v: {v:?}"); let vv = v.iter().collect::<std::collections::HashSet<_>>(); println!("vv: {vv:?}"); }
collect
๋ HashSet
์ ๊ตฌํํ FromIterator
์ ์์กดํฉ๋๋ค.
์ ์ ๋ณ์(static)๊ณผ ์์(const)
์ ์ ๋ณ์์ ์์๋ ์ ์ญ ์ค์ฝํ์์ ๊ฐ์ ์์ฑํ๋ ๋ ๊ฐ์ง ๋ฐฉ๋ฒ์ ๋๋ค. ์ ์ญ ์ค์ฝํ์ ์์ฑ๋ ๊ฐ์ ํ๋ก๊ทธ๋จ ์ํ ๋์ค์ ๋ค๋ฅธ ๊ฐ์ผ๋ก ์ด๋๋์ง ์์ผ๋ฉฐ, ๋ฉ๋ชจ๋ฆฌ ์์์ ๊ทธ ์์น๊ฐ ๋ณํ์ง ์์ต๋๋ค.
์์(const
)
์์๋ ์ปดํ์ผ ํ ๋ ๊ทธ ๊ฐ์ด ์ ํด์ง๋๋ค. ๊ทธ๋ฆฌ๊ณ ๊ทธ ๊ฐ์ ๊ทธ ์์๊ฐ ์ฌ์ฉ๋๋ ๋ชจ๋ ๋ถ๋ถ์์ ์ธ๋ผ์ธ ๋ฉ๋๋ค:
const DIGEST_SIZE: usize = 3; const ZERO: Option<u8> = Some(42); fn compute_digest(text: &str) -> [u8; DIGEST_SIZE] { let mut digest = [ZERO.unwrap_or(0); DIGEST_SIZE]; for (idx, &b) in text.as_bytes().iter().enumerate() { digest[idx % DIGEST_SIZE] = digest[idx % DIGEST_SIZE].wrapping_add(b); } digest } fn main() { let digest = compute_digest("Hello"); println!("Digest: {digest:?}"); }
Rust RFC Book์ ๋ฐ๋ฅด๋ฉด ์์๋, ๊ทธ ์์๊ฐ ์ฌ์ฉ๋๋ ๊ณณ์ ์ธ๋ผ์ธ ๋ฉ๋๋ค.
const
๊ฐ์ ์์ฑํ ๋์๋ const
๋ก ๋งํน๋ ํจ์๋ง์ด ํธ์ถ ๊ฐ๋ฅํ๋ฉฐ, ์ด ํจ์๋ค์ ์ปดํ์ผ ์์ ํธ์ถ์ด ๋ฉ๋๋ค. ๋ฌผ๋ก const
ํจ์๋ค์ ๋ฐํ์์ ํธ์ถํ๋ ๊ฒ๋ ๊ฐ๋ฅํฉ๋๋ค.
์ ์ ๋ณ์(static
)
์ ์ ๋ณ์๋ ํ๋ก๊ทธ๋จ์ด ์ํ๋๋ ๋์ ์ ์ง๊ฐ ๋ฉ๋๋ค. ๊ทธ๋ฌ๋ฏ๋ก ๋ค๋ฅธ ๋ณ์๋ก ์ด๋(move)๋์ง ์์ต๋๋ค:
static BANNER: &str = "Welcome to RustOS 3.14"; fn main() { println!("{BANNER}"); }
Rust RFC Book์์ ์ธ๊ธํ ๋ฐ์ ๊ฐ์ด, ์ ์ ๋ณ์๋ ๋ณ๋์ ๋ฉ๋ชจ๋ฆฌ ๊ณต๊ฐ์ ๊ฐ์ง๋ฉฐ, ์ธ๋ผ์ธ ๋์ง ์์ต๋๋ค. ์ ์ ๋ณ์๋ ์์ ํ์ง ์์(unsafe) ๋ฌ์คํธ์ ์๋ฒ ๋๋ ์์คํ
์ฉ ์ฝ๋์์ ์ ์ฉํฉ๋๋ค. ์ด๋ค์ ์๋ช
์ ํ๋ก๊ทธ๋จ์ด ์ํ๋๋ ์ ์ฒด ์๊ฐ๊ณผ ๋์ผํฉ๋๋ค. ์ ์ญ ์ค์ฝํ๋ฅผ ๊ฐ์ง ์ด๋ค ๊ฐ์ด, ๋ฉ๋ชจ๋ฆฌ ์์ ๋จ ํ๋๋ง ์กด์ฌํด์ผ ํ๋ค๋ ์๊ตฌ์กฐ๊ฑด์ด ์๋ค๋ฉด, ์ ์ ๋ณ์ ๋์ const
๋ฅผ ์ฐ๋ ๊ฒ์ด ์ณ์ต๋๋ค.
static
๋ณ์๋ค์ ์ด๋ค ์ค๋ ๋์์๋ ์ ๊ทผ ๊ฐ๋ฅํ๊ธฐ ๋๋ฌธ์, Sync
ํธ๋ ์์ ๊ตฌํํด์ผ ํฉ๋๋ค. ์ด ๋ณ์๋ฅผ ์ฝ๊ณ ์ฐ๋ ค๋ฉด Mutex
๋ก ๊ฐ์ธ๊ฑฐ๋, atomic ์ฐ์ฐ์ ์จ์ผ ํฉ๋๋ค. static
๋ณ์๋ฅผ mutableํ๊ฒ ์ ์ธํ ์๋ ์์ง๋ง, ์ด ๊ฒฝ์ฐ ๋๊ธฐํ ์์
์ ์๋์ผ๋ก ํด ์ฃผ์ด์ผ ํฉ๋๋ค. ๊ทธ๋์ ๊ทธ๋ฌํ ๋ณ์๋ฅผ ์ ๊ทผํ๋ ์ฝ๋๋unsafe
๋ก ๋ช
์์ ์ผ๋ก ํ์๊ฐ ๋์ด์ผ ํฉ๋๋ค. โ์์ ํ์ง ์์ ๋ฌ์คํธโ๋ฅผ ๋ฐฐ์ธ ๋ mutable statics ๋ถ๋ถ์์ ์ข ๋ ์์ธํ ๋ค๋ฃจ๊ฒ ์ต๋๋ค.
- ๋ฌ์คํธ์
const
๋ C++์constexpr
๊ณผ ๋งค์ฐ ๋น์ทํฉ๋๋ค. - ๋ฐ๋ฉด์ ๋ฌ์คํธ์
static
์ C++์const
๋ ๊ฐ๋ณ ์ ์ ๋ณ์(mutable global variable)์ ํจ์ฌ ๋ ์ ์ฌํฉ๋๋ค. static
์ ๊ฐ์ฒด์ ์ ์ฒด์ฑ์ ๋ถ์ฌํฉ๋๋ค. ์ ์ฒด์ ์ด๋ ๋ฉ๋ชจ๋ฆฌ ์์์์ ์ฃผ์, ๊ทธ๋ฆฌ๊ณ ๋ด๋ถ ์ํ๋ฅผ ์๋ฏธํฉ๋๋ค.- ํ๋ก๊ทธ๋จ ์ํ์ ๊ทธ ๊ฐ์ด ์ ํด์ง๋ ์์๊ฐ ํ์ํ ๊ฒฝ์ฐ๋ ๋๋ญ ๋๋ค. ๊ทธ๋ฌ๋ ๊ทธ๋ ๋ค๊ณ ํด๋, ์ ์ ๋ณ์๋ฅผ ์ฌ์ฉํ๋ ๊ฒ ๋ณด๋ค๋ ๋ ์ ์ฉํ๊ณ ์์ ํฉ๋๋ค.
thread_local
๋ฐ์ดํฐ๋std::thread_local
๋งคํฌ๋ก๋ฅผ ์ด์ฉํ์ฌ ์์ฑํ ์ ์์ต๋๋ค.
์์ฑ ๋น๊ต ํ ์ด๋ธ:
์์ฑ | ์ ์ (static) ๋ณ์ | ์์(constant) |
---|---|---|
๋ฉ๋ชจ๋ฆฌ ์์ ์ฃผ์๊ฐ ์๋๊ฐ | ์ | ์๋์ค(์ธ๋ผ์ธ ๋จ) |
ํ๋ก๊ทธ๋จ์ด ์ํ๋๋ ๋์ ๊ณ์ ์ด์ ์๋๊ฐ | ์ | ์๋์ค |
๋ณ๊ฒฝ ๊ฐ๋ฅํ๊ฐ | ์ (๊ทธ๋ฌ๋ ์์ ํ์ง ์์) | ์๋์ค |
์ปดํ์ผ์ ๊ทธ ๊ฐ์ด ๊ฒฐ์ ๋๋๊ฐ | ์ (์ปดํ์ผ์ ์ด๊ธฐํ ๋จ) | ์ |
์ฌ์ฉ๋๋ ๊ณณ์ ์ธ๋ผ์ธ ๋๋๊ฐ | ์๋์ค | ์ |
๋ฒ์(Scopes)์ ์๋์(Shadowing)
ํ์ฌ ๋ฒ์์ ์๋ ๋ณ์์, ๋ฐ๊นฅ ๋ฒ์์ ์๋ ๋ณ์ ๋ชจ๋ ๊ฐ๋ฆด(์๋์)์ ์์ต๋๋ค:
fn main() { let a = 10; println!("before: {a}"); { let a = "hello"; println!("inner scope: {a}"); let a = true; println!("shadowed in inner scope: {a}"); } println!("after: {a}"); }
- ์๋์์ ๊ธฐ์กด ๋ณ์์ ์๋ก์ด ๊ฐ์ ํ ๋นํ๋ ๊ฒ์ด ์๋๋๋ค. ์๋์์ ํ๋ฉด ์๋ก์ด ๋ณ์๊ฐ ์๊ธฐ๋ฉฐ, ์ด์ ๋ณ์์ ์ ๋ณ์๋ ๋ฉ๋ชจ๋ฆฌ์ ์๋ก ๋ค๋ฅธ ์์น์ ์กด์ฌํฉ๋๋ค. ๊ทธ ๋ ๋ณ์๋ ๋จ์ง ์ด๋ฆ์ด ๊ฐ์ ๋ฟ์ด๋ฉฐ, ์ฝ๋ ์ค ์ด๋์์ ๊ทธ ์ด๋ฆ์ด ์ฌ์ฉ๋์๋๋์ ๋ฐ๋ผ ์ด๋ค ๋ณ์๋ฅผ ์ง์นญํ๋ ์ง๊ฐ ๊ฒฐ์ ๋ฉ๋๋ค.
- ์๋์ ์ ํ์ ์ ๋ฐ๊ฟ ์ ์์ต๋๋ค.
- ์ฒ์์ ์๋์์ ๋ณด๋ฉด ์ฝ๋๋ฅผ ๋ ๋ชจํธํ๊ฒ ๋ง๋ ๋ค๊ณ ์๊ฐํ ์ ๋ ์์ต๋๋ค. ๊ทธ๋ฌ๋ ์ค์ ๋ก ์๋์์ ์ด์ฉํ๋ฉด, ์ด๋ค ๋ณ์์์
.unwrap()
๋ ๊ฐ์ ์๋ก์ด ๋ณ์์ ๋ด์ ๊ฒฝ์ฐ ์๋ก์ด ์ด๋ฆ์ ์ง์ ํ์ ์์ด ๊ธฐ์กด ์ด๋ฆ์ ์ ์งํ ์ ์์ด์ ํธ๋ฆฌํฉ๋๋ค. - ์๋ ์ฝ๋๋ ๋ถ๋ณ ๋ณ์๋ฅผ ์๋์ํ ๋ ํ์ ์ด ๋์ผํ๋๋ผ๋ ์ ๋ณ์๊ฐ ์๋ ๋ณ์์ ๋ฉ๋ชจ๋ฆฌ ์์น๋ฅผ ์ฌ์ฌ์ฉ ํ ์ ์๋์ง ๊ทธ ์ด์ ๋ฅผ ๋ณด์ฌ์ค๋๋ค.
fn main() { let a = 1; let b = &a; let a = a + 1; println!("{a} {b}"); }
์ด๊ฑฐํ
enum
ํค์๋๋ ๋ช๊ฐ์ง ์ ํ(variant)์ผ๋ก ํํ๋๋ ํ์
์ ์์ฑํฉ๋๋ค:
fn generate_random_number() -> i32 { // Implementation based on https://xkcd.com/221/ 4 // Chosen by fair dice roll. Guaranteed to be random. } #[derive(Debug)] enum CoinFlip { Heads, Tails, } fn flip_coin() -> CoinFlip { let random_number = generate_random_number(); if random_number % 2 == 0 { return CoinFlip::Heads; } else { return CoinFlip::Tails; } } fn main() { println!("You got: {:?}", flip_coin()); }
ํค ํฌ์ธํธ:
- ์ด๊ฑฐํ์ ๊ฐ๋ค์ ์งํฉ์ ํ๋์ ํ์ ์ผ๋ก ํํํ ์ ์๊ฒ ํฉ๋๋ค
- ์์
CoinFlip
์ด๊ฑฐํ ํ์ ์Heads
์Tail
๋ ๊ฐ์ง variant๋ฅผ ๊ฐ์ง๋๋ค. ์ด๊ฑฐํ ํ์ ์ variant๋ ๋ค์์คํ์ด์ค๋ฅผ ๋ถ์ฌ์ ์ฌ์ฉํฉ๋๋ค. - ๊ตฌ์กฐ์ฒด์ ์ด๊ฑฐํ์ ๋น๊ตํด ๋ณด๊ฒ ์ต๋๋ค:
- ๊ตฌ์กฐ์ฒด๋ ์ด๊ฑฐํ ๋ชจ๋, ํ๋๊ฐ ํ๋๋ ์๋ ๋จ์ํ ํํ๋ ๊ฐ๋ฅ ํ๊ณ , ์ฌ๋ฌ ํ์ ์ ํ๋๋ฅผ ๊ฐ์ง ์๋ ์์ต๋๋ค.
- ๋ ๋ค ์ฐ๊ดํจ์๋ฅผ
impl
๋ธ๋ก์ผ๋ก ์ ์ ํ ์ ์์ต๋๋ค. - ์ด๊ฑฐํ ํ์ ์ ๊ฐ variant๋ฅผ ๋ณ๋์ ๊ตฌ์กฐ์ฒด๋ก ์ ์ํ ์๋ ์์ง๋ง, ๊ทธ๋ฌ๋ฉด ์ด๊ฑฐํ์ ์ฌ์ฉํ์ ๋์ฒ๋ผ ํ๋์ ํ์ ์ผ๋ก ์ทจ๊ธํ ์ ์์ต๋๋ค.
๋ฐ์ดํฐ๋ฅผ ํฌํจํ๋ ์ด๊ฑฐํ(Variant Payloads)
์ข๋ ๋ณต์กํ ์ด๊ฑฐํ์ ๊ฒฝ์ฐ variant์ ๋ฐ์ดํฐ(payload)๋ฅผ ํฌํจ์ํค๋ ํฉ๋๋ค. ๊ฐ variant์ ๋ด๊ธด ๋ฐ์ดํฐ๋ match
๋ฌธ์ ์ด์ฉํด ์ถ์ถํฉ๋๋ค:
enum WebEvent { PageLoad, // Variant without payload KeyPress(char), // Tuple struct variant Click { x: i64, y: i64 }, // Full struct variant } #[rustfmt::skip] fn inspect(event: WebEvent) { match event { WebEvent::PageLoad => println!("page loaded"), WebEvent::KeyPress(c) => println!("pressed '{c}'"), WebEvent::Click { x, y } => println!("clicked at x={x}, y={y}"), } } fn main() { let load = WebEvent::PageLoad; let press = WebEvent::KeyPress('x'); let click = WebEvent::Click { x: 20, y: 80 }; inspect(load); inspect(press); inspect(click); }
- ์ด๊ฑฐํ ์์ ๊ฐ์ ํจํด ๋งค์นญ์ด ๋๊ณ ๋ ์ดํ์๋ง ์ ๊ทผ ๊ฐ๋ฅํฉ๋๋ค. ๊ทธ ๊ฐ์ ๋ํ ๋ ํผ๋ฐ์ค๋
=>
์ดํ์ ์ฌ์ฉ๊ฐ๋ฅํฉ๋๋ค.- ๋งค์น ํจํด๋ค์ ์์์ ์๋๋ก ์์์ ๋ฐ๋ผ ๊ฒ์ฌํฉ๋๋ค. C๋ C++์์์ ๊ฐ์ fall-through๋ ์์ต๋๋ค.
- ๋งค์น ํํ์ ์์ฒด๋ ๊ฐ์ ๊ฐ์ง๋๋ค. ๊ทธ ๊ฐ์ ๋งค์นญ์ด ๋ ํจํด์์ ๊ฐ์ฅ ๋ง์ง๋ง์ ์ํ๋ ํํ์์ด ๋ฉ๋๋ค.
- ๊ฐ์ฅ ์์์ ๋ถํฐ ์ด๋ค ํจํด์ด ์ฃผ์ด์ง ๊ฐ๊ณผ ๋งค์นญํ๋์ง ๊ฒ์ฌํ ๋ค์, ๋งค์นญ๋ ๊ฒ์ด ๋ฐ๊ฒฌ๋๋ฉด ํ์ดํ๋ฅผ ๋ฐ๋ผ ์ฝ๋๋ฅผ ์ํํฉ๋๋ค. ํ ๋ฒ ๋งค์นญ์ด ๋๊ณ ์ฝ๋๊ฐ ์ํ์ด ๋๋ฉด, ๋์ด์์ ๋งค์นญ์ ์์ต๋๋ค.
- ๋งค์นญ ํจํด๋ค์ด ๋ถ์ถฉ๋ถ ํ๋ค๋ฉด ์ด๋ค ์ผ์ด ์ผ์ด๋๋์ง ์ค๋ช ํ์ธ์. ๋ฌ์คํธ ์ปดํ์ผ๋ฌ๋ ๋ชจ๋ ๊ฐ๋ฅํ ์ผ์ด์ค๋ค์ด ํธ๋ค๋ง ๋๋์ง ์ฒดํฌํ๋ค๋ ์ ์ ์๊ธฐ์ํค์ธ์.
match
๋ ์ฃผ์ด์ง ์ด๊ฑฐํ ๊ฐ์ด ์ค์ ๋ก ์ด๋ค variant์ธ์ง ํ๋จํ๊ธฐ ์ํด, ๊ทธ variant์ ์ข ๋ฅ๊ฐ ๊ธฐ๋ก๋, ์จ๊ฒจ์ง ํ๋(์๋ณ์)์ ๊ฐ์ ๊ฒ์ฌํฉ๋๋ค.std::mem::discriminant()
๋ฅผ ์ด์ฉํ์ฌ ์๋ณ์๋ฅผ ์ป์ ์๋ ์์ต๋๋ค- ์ด๋ ๊ฐ ํ๋ ๊ฐ์ ๊ตณ์ด ๋น๊ตํ ํ์ ์๋ ๊ตฌ์กฐ์ฒด์ ๋ํด
PartialEq
ํธ๋ ์์ ๊ตฌํํ ๋ ์ ์ฉํฉ๋๋ค.
- ์ด๋ ๊ฐ ํ๋ ๊ฐ์ ๊ตณ์ด ๋น๊ตํ ํ์ ์๋ ๊ตฌ์กฐ์ฒด์ ๋ํด
WebEvent::Click { ... }
์ ์ต์์ ๋ ๋ฒจ ๊ตฌ์กฐ์ฒดstruct Click {...}
๋ฅผ ๋ฐ๋ก ์ ์ํ๊ณWebEvent::Click(Click)
์ฒ๋ผ ํํ ํํ๋ก ์ ์ํ ๊ฒ๊ณผ ์ ํํ ๊ฐ์ง ์์ต๋๋ค. ์๋ฅผ ๋ค์ดWebEvent::Click { ... }
๋ก ์ ์ํ ๊ฒฝ์ฐ, ๊ตฌ์กฐ์ฒด ํํ์ ์ ์ฌํ์ง๋ง ํธ๋ ์์ ๊ตฌํ ํ ์ ์์ต๋๋ค.
์ด๊ฑฐํ์ ํฌ๊ธฐ
๋ฌ์คํธ์ ์ด๊ฑฐํ์ ์ ๋ ฌ(alignment)๋ก ์ธํ ์ ์ฝ์ ๊ณ ๋ คํ์ฌ ํฌ๊ธฐ๋ฅผ ๋นฝ๋นฝํ๊ฒ ์ก์ต๋๋ค:
use std::any::type_name; use std::mem::{align_of, size_of}; fn dbg_size<T>() { println!("{}: size {} bytes, align: {} bytes", type_name::<T>(), size_of::<T>(), align_of::<T>()); } enum Foo { A, B, } fn main() { dbg_size::<Foo>(); }
- ์์ธํ ์ฌํญ์ ๊ณต์๋ฌธ์๋ฅผ ํ์ธํ์ธ์.
ํค ํฌ์ธํธ:
-
๋ฌ์คํธ๋ ์ด๊ฑฐํ variant๋ฅผ ๊ตฌ๋ถํ๊ธฐ ์ํด ๋ด๋ถ์ ์ผ๋ก ์๋ณ์(discriminant) ํ๋๋ฅผ ์ฌ์ฉํฉ๋๋ค.
-
C์์ ์ฐ๋์ ์ํด ์๋ณ์ ๊ฐ์ ์ง์ ์ง์ ํ ์๋ ์์ต๋๋ค:
#[repr(u32)] enum Bar { A, // 0 B = 10000, C, // 10001 } fn main() { println!("A: {}", Bar::A as u32); println!("B: {}", Bar::B as u32); println!("C: {}", Bar::C as u32); }
repr
์์ฑ์ด ์๋ค๋ฉด 10001์ด 2 ๋ฐ์ดํธ๋ก ํํ๊ฐ๋ฅํ๊ธฐ ๋๋ฌธ์ ์๋ณ์์ ํ์ ํฌ๊ธฐ๋ 2 ๋ฐ์ดํธ๊ฐ ๋ฉ๋๋ค. -
๋ค๋ฅธ ํ์ ๋ค๋ ํ์ธํด๋ณด์ธ์
dbg_size!(bool)
: ํฌ๊ธฐ 1 ๋ฐ์ดํธ, ์ ๋ ฌ: 1 ๋ฐ์ดํธ,dbg_size!(Option<bool>)
: ํฌ๊ธฐ 1 ๋ฐ์ดํธ, ์ ๋ ฌ: 1 ๋ฐ์ดํธ (๋์น ์ต์ ํ, ์๋ ์ค๋ช ์ฐธ์กฐ)dbg_size!(&i32)
: ํฌ๊ธฐ 8 ๋ฐ์ดํธ, ์ ๋ ฌ: 8 ๋ฐ์ดํธ (64๋นํธ ๋จธ์ ์ธ ๊ฒฝ์ฐ)dbg_size!(Option<&i32>)
: ํฌ๊ธฐ 8 ๋ฐ์ดํธ, ์ ๋ ฌ: 8 ๋ฐ์ดํธ (๋ํฌ์ธํฐ ์ต์ ํ, ์๋ ์ค๋ช ์ฐธ์กฐ)
-
๋์น ์ต์ ํ: ๋ฌ์คํธ๋ ์ด๊ฑฐํ ์๋ณ์๋ฅผ ์ฌ์ฉ๋์ง ์์ ๋นํธ ํจํด๊ณผ ๋ณํฉํฉ๋๋ค.
-
๋ํฌ์ธํฐ ์ต์ ํ: ์ด๋ค ํ์ ๋ค์ ๋ํด์ ๋ฌ์คํธ๋
size_of::<T>()
๊ฐsize_of::<Option<T>>()
์ ๊ฐ์ ๊ฒ์ ๋ณด์ฅํฉ๋๋ค.์ค์ ๋ก ๋ํฌ์ธํฐ ์ต์ ํ๊ฐ ์ ์ฉ๋ ๊ฒ์ ํ์ธํ๊ณ ์ถ๋ค๋ฉด ์๋์ ์์ ์ฝ๋๋ฅผ ์ฌ์ฉํ์ธ์. ์ฃผ์ํ ์ ์, ์ฌ๊ธฐ์์ ๋ณด์ฌ์ฃผ๋ ๋นํธ ํจํด์ด ์ปดํ์ผ๋ฌ๊ฐ ๋ณด์ฅํด ์ฃผ๋ ๊ฒ์ ์๋๋ผ๋ ์ ์ ๋๋ค. ์ฌ๊ธฐ์ ์์กดํ๋ ๊ฒ์ ์์ ํ unsafeํฉ๋๋ค.
use std::mem::transmute; macro_rules! dbg_bits { ($e:expr, $bit_type:ty) => { println!("- {}: {:#x}", stringify!($e), transmute::<_, $bit_type>($e)); }; } fn main() { // TOTALLY UNSAFE. Rust provides no guarantees about the bitwise // representation of types. unsafe { println!("Bitwise representation of bool"); dbg_bits!(false, u8); dbg_bits!(true, u8); println!("Bitwise representation of Option<bool>"); dbg_bits!(None::<bool>, u8); dbg_bits!(Some(false), u8); dbg_bits!(Some(true), u8); println!("Bitwise representation of Option<Option<bool>>"); dbg_bits!(Some(Some(false)), u8); dbg_bits!(Some(Some(true)), u8); dbg_bits!(Some(None::<bool>), u8); dbg_bits!(None::<Option<bool>>, u8); println!("Bitwise representation of Option<&i32>"); dbg_bits!(None::<&i32>, usize); dbg_bits!(Some(&0i32), usize); } }
์๋๋ ์ข ๋ ๋ณต์กํ ์์ ์ ๋๋ค. 256๊ฐ ์ด์์
Option
์ด ์ค์ฒฉ๋์ด ์ฐ๊ฒฐ๋์ด ์์ ๊ฒฝ์ฐ ์ด๋ป๊ฒ ๋๋์ง ๋ณด์ฌ์ค๋๋ค.#![recursion_limit = "1000"] use std::mem::transmute; macro_rules! dbg_bits { ($e:expr, $bit_type:ty) => { println!("- {}: {:#x}", stringify!($e), transmute::<_, $bit_type>($e)); }; } // Macro to wrap a value in 2^n Some() where n is the number of "@" signs. // Increasing the recursion limit is required to evaluate this macro. macro_rules! many_options { ($value:expr) => { Some($value) }; ($value:expr, @) => { Some(Some($value)) }; ($value:expr, @ $($more:tt)+) => { many_options!(many_options!($value, $($more)+), $($more)+) }; } fn main() { // TOTALLY UNSAFE. Rust provides no guarantees about the bitwise // representation of types. unsafe { assert_eq!(many_options!(false), Some(false)); assert_eq!(many_options!(false, @), Some(Some(false))); assert_eq!(many_options!(false, @@), Some(Some(Some(Some(false))))); println!("Bitwise representation of a chain of 128 Option's."); dbg_bits!(many_options!(false, @@@@@@@), u8); dbg_bits!(many_options!(true, @@@@@@@), u8); println!("Bitwise representation of a chain of 256 Option's."); dbg_bits!(many_options!(false, @@@@@@@@), u16); dbg_bits!(many_options!(true, @@@@@@@@), u16); println!("Bitwise representation of a chain of 257 Option's."); dbg_bits!(many_options!(Some(false), @@@@@@@@), u16); dbg_bits!(many_options!(Some(true), @@@@@@@@), u16); dbg_bits!(many_options!(None::<bool>, @@@@@@@@), u16); } }
Novel Control Flow
Rust has a few control flow constructs which differ from other languages. They are used for pattern matching:
if let
ํํ์while let
expressionsmatch
ํํ์
if let
ํํ์
if let
ํํ์์ ์ฌ์ฉํ๋ฉด ๊ฐ์ด ํจํด๊ณผ ์ผ์นํ๋์ง์ ๋ฐ๋ผ ๋ค๋ฅธ ์ฝ๋๋ฅผ ์คํํ ์ ์์ต๋๋ค:
fn main() { let arg = std::env::args().next(); if let Some(value) = arg { println!("Program name: {value}"); } else { println!("Missing name?"); } }
ํจํด์ ๊ดํ ์ค๋ช ์ ํจํด ๋งค์นญ์ ์ฐธ์กฐํ์ธ์.
-
if let
์ดmatch
๋ณด๋ค ๋ ๊ฐ๊ฒฐํ ์ ์์ต๋๋ค(์: ํ๊ฐ์ง ๋ธ๋์น๋ง ํฅ๋ฏธ๋ก์ด ๊ฒฝ์ฐ). ์ด์ ๋ฌ๋ฆฌmatch
์์๋ ๋ชจ๋ ๋ธ๋์น๊ฐ ์ฒ๋ฆฌ๋์ด์ผ ํฉ๋๋ค. -
์ผ๋ฐ์ ์ฌ์ฉ๋ฒ์
Option
์ ์ฌ์ฉํ ๋Some
๊ฐ์ ์ฒ๋ฆฌํ๋ ๊ฒ์ ๋๋ค. -
match
์ ๋ฌ๋ฆฌif let
์ ํจํด ์ผ์น๋ฅผ ์ํ ๋ณดํธ ์ ์ ์ง์ํ์ง ์์ต๋๋ค. -
1.65๋ถํฐ ์ ์ฌํ let-else ๊ตฌ์ฑ์ ๋์คํธ๋ญ์ฒ๋ง ํ ๋น์ ์คํํ๊ฑฐ๋ ์คํจํ ๊ฒฝ์ฐ ๋ฐํ๋์ง ์๋ ๋ธ๋ก ๋ธ๋์น(panic/return/break/continue)๋ฅผ ๋ณด์ ํ๋๋ก ํ์ฉํฉ๋๋ค:
fn main() { println!("{:?}", second_word_to_upper("foo bar")); } fn second_word_to_upper(s: &str) -> Option<String> { let mut it = s.split(' '); let (Some(_), Some(item)) = (it.next(), it.next()) else { return None; }; Some(item.to_uppercase()) }
while let
๋ฐ๋ณต๋ฌธ
๋ง์ง๋ง์ผ๋ก, ๋ฌดํ ๋ฃจํ๋ฅผ ๋ง๋๋ loop
ํค์๋๊ฐ ์์ต๋๋ค:
fn main() { let v = vec![10, 20, 30]; let mut iter = v.into_iter(); while let Some(x) = iter.next() { println!("x: {x}"); } }
v.into_iter()
๊ฐ ๋ฐํํ ๋ฐ๋ณต์๋ next()
๊ฐ ํธ์ถ๋ ๋๋ง๋ค Option<i32>
๋ฅผ ๋ฐํํฉ๋๋ค. ๋ฐ๋ณต์๊ฐ ์๋ฃ๋ ๋๊น์ง๋ Some(x)
๋ฅผ ๋ฐํํ๊ณ ๋ง์ง๋ง์ None
์ ๋ฐํํฉ๋๋ค. while let
์ ํตํด ๋ฐ๋ณต์์ ๋ชจ๋ ์์ดํ
์ ํ์ธํ ์ ์์ต๋๋ค.
ํจํด์ ๊ดํ ์ค๋ช ์ ํจํด ๋งค์นญ์ ์ฐธ์กฐํ์ธ์.
while let
์ ๊ฐ์ด ํจํด์ ๋งค์น๋๋ ๋์ ๊ณ์๋ฉ๋๋ค.while let
๋ฃจํ ๋์ ๋ฌดํ ๋ฃจํ๋ฅผ ์ฌ์ฉํ๊ณiter.next()
๊ฐ ๋น ๊ฐ์ ๋ฐํํ ๋ ๋ฃจํ๋ฅผ ๋น ์ ธ๋์ค๋๋ก ์์ฑํ ์๋ ์์ต๋๋ค.while let
์ ๊ทธ๋ฌํ ๊ฒฝ์ฐ๋ฅผ ์ํ ๋ฌธ๋ฒ์ ํธ์๋ฅผ ์ ๊ณตํฉ๋๋ค.
match
ํํ์
match
ํค์๋๋ ์ด๋ค ๊ฐ์ ํ๋ ์ด์์ ํจํด์ ๋ํด ๋งค์นํ๋๋ฐ ์ฌ์ฉํฉ๋๋ค. ๊ทธ๋ฐ ๋ฉด์์ if let
ํํ์์ ์ฌ๋ฌ๊ฐ ์ด์ด ๋์ ๊ฒ๊ณผ ๊ฐ์ต๋๋ค:
fn main() { match std::env::args().next().as_deref() { Some("cat") => println!("Will do cat things"), Some("ls") => println!("Will ls some files"), Some("mv") => println!("Let's move some files"), Some("rm") => println!("Uh, dangerous!"), None => println!("Hmm, no program name?"), _ => println!("Unknown program name!"), } }
if let
๊ณผ ๋ง์ฐฌ๊ฐ์ง๋ก ๋งค์น์ ๋ชจ๋ ํ(arm)์ ๊ฐ์ ํ์
์ด์ด์ผ ํฉ๋๋ค. ํ์ด ๋ธ๋ก์ด๋ผ๋ฉด ๋ธ๋ก์ ๋ง์ง๋ง ํํ์์ด ๊ทธ ํ์
์ด ๋ฉ๋๋ค. ์์ ์์ ์์ ๋งค์น ํํ์์ ํ์
์ ()
์
๋๋ค.
ํจํด์ ๊ดํ ์ค๋ช ์ ํจํด ๋งค์นญ์ ์ฐธ์กฐํ์ธ์.
match
ํํ์์ ๋ณ์์ ํ ๋นํ๊ณ ๊ทธ ๊ฐ์ ์ถ๋ ฅํด๋ณด์ธ์..as_deref()
๋ฅผ ์ง์๋ณด๊ณ , ์ด ๋ ๋์ค๋ ์๋ฌ๋ฅผ ์ค๋ช ํด์ฃผ์ธ์.std::env::args().next()
๋Option<String>
๊ฐ์ ๋ฐํํ๋๋ฐ,String
์ ์ง์ ๋งค์นํ ์ ์์ต๋๋ค.as_deref()
๋Option<T>
๋ฅผOption<&T::Target>
์ผ๋ก ๋ฐ๊ฟ์ค๋๋ค. ์ด ๊ฒฝ์ฐ๋Option<String>
์์Option<&str>
๋ก ๋ฐ๋๋๋ค.- ์ด์ ๋ ํจํด ๋งค์นญ์ผ๋ก
Option
์์&str
์ ๋งค์นํ ์ ์์ต๋๋ค.
ํจํด ๋งค์นญ
match
ํค์๋๋ ๊ฐ์ ์ฌ๋ฌ ํํ์ ํจํด๊ณผ ๋งค์น์ํฌ ์ ์์ต๋๋ค. ๋งจ ์ ํจํด๋ถํฐ ํ๋์ฉ ๋งค์น๋๋์ง ๊ฒ์ฌํ๋ฉฐ, ์ฒ์์ผ๋ก ๋งค์น๋๋ ํจํด์ด ์ ํ๋ฉ๋๋ค.
C/C++์ switch
์ ๋น์ทํ๊ฒ ๊ฐ์ ํจํด์ผ๋ก ์ฌ์ฉํ ์๋ ์์ต๋๋ค:
fn main() { let input = 'x'; match input { 'q' => println!("Quitting"), 'a' | 's' | 'w' | 'd' => println!("Moving around"), '0'..='9' => println!("Number input"), _ => println!("Something else"), } }
_
ํจํด์ ์ด๋ค ๊ฐ๊ณผ๋ ๋งค์นญ๋๋ ์์ผ๋์นด๋์
๋๋ค.
ํค ํฌ์ธํธ:
- ํจํด์์ ์ฌ์ฉ๋๋ ํน์ ๋ฌธ์๋ค์ ์๋ ค์ฃผ์ธ์
|
: or ๊ธฐํธ์ ๋๋ค..
: ํ์ํ ๋งํผ ํ์ฅํฉ๋๋ค1..=5
: ๋ ๊ฐ(์ฌ๊ธฐ์๋ 5)์ ํฌํจํ๋ ๋ฒ์๋ฅผ ๋ํ๋ ๋๋ค_
: ์์ผ๋์นด๋์ ๋๋ค
- ์์ผ๋์นด๋ ๋ฌธ์๋ฅผ ๋ณ์๋ก ๋ฐ๊พธ๊ฑฐ๋
q
์ ๋ฐ์ดํ๋ฅผ ์ ๊ฑฐํ๋ ์์ผ๋ก ์์ ํ๋ฉด์ ๋ฐ์ธ๋ฉ์ด ์ด๋ป๊ฒ ์๋ํ๋์ง ๋ณด์ฌ์ฃผ๋ ๊ฒ๋ ์ ์ฉํ ์ ์์ต๋๋ค. - ์ฐธ์กฐ๋ฅผ ๋งค์นญํ๋ ๊ฒ๋ ์์ฐํ ์ ์์ต๋๋ค.
- ์๋ฌ ๋ฉ์์ง์ โ๋ฐ๋ฐ ๋ถ๊ฐ๋ฅ ํจํด(irrefutable pattern)โ์ด๋ ์ฉ์ด๊ฐ ๋ฑ์ฅํ๊ธฐ๋ ํฉ๋๋ค. ์ง๊ธ ๊ทธ ์๋ฏธ๋ฅผ ์๊ฐํ๋ ๊ฒ๋ ์ข์ ๊ฒ ๊ฐ์ต๋๋ค.
์ด๊ฑฐํ ๋ถํด(์ญ๊ตฌ์กฐํ)
๊ตฌ์กฐ์ฒด๋ ์ด๊ฑฐํ ๊ฐ์ ์ผ๋ถ๋ฅผ ํจํด ๋งค์น๋ฅผ ํตํด ๋ณ์์ ๋ฐ์ธ๋ฉํ ์ ์์ต๋๋ค. ๊ฐ๋จํ enum
ํ์
์ ๋จผ์ ์ดํด๋ณด๊ฒ ์ต๋๋ค:
enum Result { Ok(i32), Err(String), } fn divide_in_two(n: i32) -> Result { if n % 2 == 0 { Result::Ok(n / 2) } else { Result::Err(format!("cannot divide {n} into two equal parts")) } } fn main() { let n = 100; match divide_in_two(n) { Result::Ok(half) => println!("{n} divided in two is {half}"), Result::Err(msg) => println!("sorry, an error happened: {msg}"), } }
match
๊ตฌ๋ฌธ์์ divide_in_two
ํจ์์์ ๋ฐํ๋๋ Result
๊ฐ์ ๋ ๊ฐ์ ํ(ํน์ ๊ฐ์ง)๋ก ๋ถํด(destructure) ํ์์ต๋๋ค. ์ฒซ๋ฒ์งธ ํ์์ half
๋ Ok
variant์ ๋ด๊ธด ๊ฐ์ผ๋ก ๋ฐ์ธ๋ฉ๋ฉ๋๋ค. ๋๋ฒ์งธ ํ์์ msg
๋ ์ค๋ฅ ๋ฉ์์ง ๋ฌธ์์ด์ ๋ฐ์ธ๋ฉ๋ฉ๋๋ค.
ํค ํฌ์ธํธ:
if
/else
ํํ์์ ์ด๊ฑฐํ์ ๋ฐํํ๊ณ , ์ด ๊ฐ์ ๋์ค์match
๋ก ๋ถํด๋ฉ๋๋ค.- ์ด๊ฑฐํ์ ์ธ๋ฒ์งธ variant๋ฅผ ์ถ๊ฐํ๊ณ ์ฝ๋๋ฅผ ์คํํ์ฌ ์ค๋ฅ๋ฅผ ํ์ํด๋ณด์ธ์. ์ฝ๋ ์ด๋ ๋ถ๋ถ์ ๋๋ฝ์ด ์๋์ง, ๊ทธ๋ฆฌ๊ณ ์ปดํ์ผ๋ฌ๊ฐ ์ด๋ค ์์ผ๋ก ํํธ๋ฅผ ์ฃผ๋์ง ๊ฐ์ด ์ดํด๋ณด์ธ์.
๊ตฌ์กฐ์ฒด ๋ถํด(์ญ๊ตฌ์กฐํ)
struct
๊ตฌ์กฐ์ฒด ์ญ์ ๋ถํดํ ์ ์์ต๋๋ค:
struct Foo { x: (u32, u32), y: u32, } #[rustfmt::skip] fn main() { let foo = Foo { x: (1, 2), y: 3 }; match foo { Foo { x: (1, b), y } => println!("x.0 = 1, b = {b}, y = {y}"), Foo { y: 2, x: i } => println!("y = 2, x = {i:?}"), Foo { y, .. } => println!("y = {y}, other fields were ignored"), } }
foo
์ ๋ฆฌํฐ๋ด ๊ฐ์ ๋ค๋ฅธ ํจํด๊ณผ ์ผ์นํ๋๋ก ๋ณ๊ฒฝํฉ๋๋ค.Foo
์ ์ ํ๋๋ฅผ ์ถ๊ฐํ๊ณ ํ์์ ๋ฐ๋ผ ํจํด์ ๋ณ๊ฒฝํฉ๋๋ค.- ์บก์ฒ์ ์์ ํํ์์ ๊ตฌ๋ถํ๊ธฐ ์ด๋ ค์ธ ์ ์์ต๋๋ค. ๋ ๋ฒ์งธ ๋ถ๋ฌธ์
2
๋ฅผ ๋ณ์๋ก ๋ณ๊ฒฝํด ๋ณด๊ณ ์๋ํ์ง ์๋ ๊ฒ์ ํ์ธํ์ธ์.const
๋ก ๋ณ๊ฒฝํ๊ณ ๋ค์ ์๋ํ๋์ง ํ์ธํฉ๋๋ค.
๋ฐฐ์ด ๋ถํด(์ญ๊ตฌ์กฐํ)
๋ฐฐ์ด์ด๋ ํํ, ์ฌ๋ผ์ด์ค๋ ๊ทธ ์์๋ค์ ๋ํด ํจํด ๋งค์นญ์ผ๋ก ๋ถํดํ ์ ์์ต๋๋ค:
#[rustfmt::skip] fn main() { let triple = [0, -2, 3]; println!("Tell me about {triple:?}"); match triple { [0, y, z] => println!("First is 0, y = {y}, and z = {z}"), [1, ..] => println!("First is 1 and the rest were ignored"), _ => println!("All elements were ignored"), } }
-
๊ธธ์ด๋ฅผ ์ ์ ์๋ ์ฌ๋ผ์ด์ค์ ๋ํด์๋ ๊ณ ์ ๊ธธ์ด ํจํด์ผ๋ก ๋ถํดํ ์ ์์ต๋๋ค.
fn main() { inspect(&[0, -2, 3]); inspect(&[0, -2, 3, 4]); } #[rustfmt::skip] fn inspect(slice: &[i32]) { println!("Tell me about {slice:?}"); match slice { &[0, y, z] => println!("First is 0, y = {y}, and z = {z}"), &[1, ..] => println!("First is 1 and the rest were ignored"), _ => println!("All elements were ignored"), } }
-
_
๋ฅผ ์ฌ์ฉํ์ฌ ์์๋ฅผ ๋งค์นญํ๋ ํจํด์ ์ถ๊ฐํด๋ณด์ธ์. -
๋ฐฐ์ด์ ๊ฐ์ ๋ ์ถ๊ฐํด๋ณด์ธ์.
-
..
๊ฐ ์์ ๊ฐ์์ ์๊ด์์ด ๋งค์น๋ ์ ์์์ ์๋ ค์ฃผ์ธ์. -
[.., b]
๋[a@.., b]
์ ๊ฐ์ ํจํด์ผ๋ก ๊ผฌ๋ฆฌ ๋ถ๋ถ์ ๋งค์นญํ๋ ๊ฒ์ ๋ณด์ฌ์ฃผ์ธ์
๋งค์น ๊ฐ๋
ํจํด ๋ค์ ๊ฐ๋(guard, ์กฐ๊ฑด์)๋ฅผ ๋ง๋ถ์ผ ์ ์์ต๋๋ค. ๊ฐ๋๋ ํจํด์ด ๋งค์น๋๋ฉด ์ถ๊ฐ๋ก ๋ฐ์ ธ๋ณด๋ ๋ถ๋ฆฌ์ธ ํํ์์ ๋๋ค:
#[rustfmt::skip] fn main() { let pair = (2, -2); println!("Tell me about {pair:?}"); match pair { (x, y) if x == y => println!("These are twins"), (x, y) if x + y == 0 => println!("Antimatter, kaboom!"), (x, _) if x % 2 == 1 => println!("The first one is odd"), _ => println!("No correlation..."), } }
ํค ํฌ์ธํธ:
- ๋งค์น ๊ฐ๋๋ ๋ณ๋์ ๋ฌธ๋ฒ ์์๋ก์ ํจํด ์์ฒด๋ง์ผ๋ก ํํํ๊ธฐ ์ด๋ ค์ด ๋ณต์กํ ๊ฒฝ์ฐ๋ฅผ ๊ฐ๊ฒฐํ๊ฒ ํํํ๊ณ ์ ํ ๋ ์ ์ฉํฉ๋๋ค.
- ๋งค์น์ ๊ฐ ํ(ํน์ ๊ฐ์ง) ์์ ๋ฐ๋ก
if
๋ฅผ ์ฌ์ฉํ ๊ฒ๊ณผ ๋ค๋ฆ ๋๋ค. ๋งค์น ๊ฐ์ง์=>
๋ค์ ์ฌ์ฉ๋if
ํํ์์ ํด๋น ๊ฐ์ง๊ฐ ์ ํ๋ ๋ค์์ ์คํ๋ฉ๋๋ค. ๋ฐ๋ผ์ ์ฌ๊ธฐ์if
์กฐ๊ฑด์ด ์คํจํ๋๋ผ๋ ์๋match
์ ๋ค๋ฅธ ๊ฐ์ง๋ ๊ณ ๋ ค๋์ง ์์ต๋๋ค. - ํจํด์ ์ ์๋ ๋ณ์๋ฅผ ๊ฐ๋์ ํํ์์์ ์ฌ์ฉํ ์ ์์ต๋๋ค.
- ๊ฐ๋์ ์ ์๋ ์กฐ๊ฑด์
|
๋ฅผ ํฌํจํ๋ ํจํด์ ๋ชจ๋ ํํ์์ ์ ์ฉ๋ฉ๋๋ค.
1์ผ์ฐจ ์คํ ์ฐ์ต๋ฌธ์
์ด๋ฒ ์ฐ์ต๋ฌธ์ ๋ ์๋ ๋๊ฐ์ง์ ๋๋ค:
-
The Luhn algorithm,
-
An exercise on pattern matching.
After looking at the exercises, you can look at the solutions provided.
๋ฃฌ ์๊ณ ๋ฆฌ์ฆ
๋ฃฌ(Luhn) ์๊ณ ๋ฆฌ์ฆ์ ์ ์ฉ์นด๋ ๋ฒํธ ๊ฒ์ฆ์ ์ฌ์ฉ๋๋ ์๊ณ ๋ฆฌ์ฆ ์
๋๋ค. ์ด ์๊ณ ๋ฆฌ์ฆ์ ์ ์ฉ์นด๋ ๋ฒํธ๋ฅผ ๋ฌธ์์ด
๋ก ์
๋ ฅ๋ฐ๊ณ , ์๋์ ์์์ ๋ฐ๋ผ ์ ์ฉ์นด๋ ๋ฒํธ์ ์ ํจ์ฑ์ ํ์ธํฉ๋๋ค:
-
๋ชจ๋ ๊ณต๋ฐฑ์ ๋ฌด์ํฉ๋๋ค. 2์๋ฆฌ ๋ฏธ๋ง ์ซ์๋ ๋ฌด์ํฉ๋๋ค.
-
์ค๋ฅธ์ชฝ์์ ์ผ์ชฝ์ผ๋ก ์ด๋ํ๋ฉฐ 2๋ฒ์งธ ์๋ฆฌ๋ง๋ค ์ซ์๋ฅผ 2๋ฐฐ ์ฆ๊ฐ์ํต๋๋ค. ์๋ฅผ ๋ค์ด
1234
์์3
๊ณผ1
์ ๊ฐ๊ฐ 2๋ฅผ ๊ณฑํฉ๋๋ค. -
After doubling a digit, sum the digits if the result is greater than 9. So doubling
7
becomes14
which becomes1 + 4 = 5
. -
๋ชจ๋ ์๋ฆฌ์ ์ซ์๋ฅผ ๋ํฉ๋๋ค.
-
ํฉ๊ณ์ ๋์๋ฆฌ๊ฐ
0
์ธ ๊ฒฝ์ฐ ์ ํจํ ์ ์ฉ์นด๋ ๋ฒํธ์ ๋๋ค.
์๋ ์ฝ๋๋ฅผ https://play.rust-lang.org/์ ๋ณต์ฌํด์ ๊ตฌํํ์๋ฉด ๋ฉ๋๋ค.
for
๋ฐ๋ณต๋ฌธ๊ณผ ์ธ๋ฑ์ค๋ฅผ ์ด์ฉํ๋ โ์ฌ์ดโ๋ฐฉ๋ฒ์ผ๋ก ๋จผ์ ํ์ด ๋ณด์ธ์. ๊ทธ๋ฐ ๋ค์ ๋ฐ๋ณต์๋ฅผ ์ด์ฉํด์ ๋ค์ ํ์ด ๋ณด์ธ์.
// TODO: remove this when you're done with your implementation. #![allow(unused_variables, dead_code)] pub fn luhn(cc_number: &str) -> bool { unimplemented!() } #[test] fn test_non_digit_cc_number() { assert!(!luhn("foo")); assert!(!luhn("foo 0 0")); } #[test] fn test_empty_cc_number() { assert!(!luhn("")); assert!(!luhn(" ")); assert!(!luhn(" ")); assert!(!luhn(" ")); } #[test] fn test_single_digit_cc_number() { assert!(!luhn("0")); } #[test] fn test_two_digit_cc_number() { assert!(luhn(" 0 0 ")); } #[test] fn test_valid_cc_number() { assert!(luhn("4263 9826 4026 9299")); assert!(luhn("4539 3195 0343 6467")); assert!(luhn("7992 7398 713")); } #[test] fn test_invalid_cc_number() { assert!(!luhn("4223 9826 4026 9299")); assert!(!luhn("4539 3195 0343 6476")); assert!(!luhn("8273 1232 7352 0569")); } #[allow(dead_code)] fn main() {}
Exercise: Expression Evaluation
Letโs write a simple recursive evaluator for arithmetic expressions.
#![allow(unused)] fn main() { /// An operation to perform on two subexpressions. #[derive(Debug)] enum Operation { Add, Sub, Mul, Div, } /// An expression, in tree form. #[derive(Debug)] enum Expression { /// An operation on two subexpressions. Op { op: Operation, left: Box<Expression>, right: Box<Expression>, }, /// A literal value Value(i64), } /// The result of evaluating an expression. #[derive(Debug, PartialEq, Eq)] enum Res { /// Evaluation was successful, with the given result. Ok(i64), /// Evaluation failed, with the given error message. Err(String), } // Allow `Ok` and `Err` as shorthands for `Res::Ok` and `Res::Err`. use Res::{Err, Ok}; fn eval(e: Expression) -> Res { todo!() } #[test] fn test_value() { assert_eq!(eval(Expression::Value(19)), Ok(19)); } #[test] fn test_sum() { assert_eq!( eval(Expression::Op { op: Operation::Add, left: Box::new(Expression::Value(10)), right: Box::new(Expression::Value(20)), }), Ok(30) ); } #[test] fn test_recursion() { let term1 = Expression::Op { op: Operation::Mul, left: Box::new(Expression::Value(10)), right: Box::new(Expression::Value(9)), }; let term2 = Expression::Op { op: Operation::Mul, left: Box::new(Expression::Op { op: Operation::Sub, left: Box::new(Expression::Value(3)), right: Box::new(Expression::Value(4)), }), right: Box::new(Expression::Value(5)), }; assert_eq!( eval(Expression::Op { op: Operation::Add, left: Box::new(term1), right: Box::new(term2), }), Ok(85) ); } #[test] fn test_error() { assert_eq!( eval(Expression::Op { op: Operation::Div, left: Box::new(Expression::Value(99)), right: Box::new(Expression::Value(0)), }), Err(String::from("division by zero")) ); } }
The Box
type here is a smart pointer, and will be covered in detail later in the course. An expression can be โboxedโ with Box::new
as seen in the tests. To evaluate a boxed expression, use the deref operator to โunboxโ it: eval(*boxed_expr)
.
Some expressions cannot be evaluated and will return an error. The Res
type represents either a successful value or an error with a message. This is very similar to the standard-library Result
which we will see later.
Copy and paste the code into the Rust playground, and begin implementing eval
. The final product should pass the tests. It may be helpful to use todo!()
and get the tests to pass one-by-one.
If you finish early, try writing a test that results in an integer overflow. How could you handle this with Res::Err
instead of a panic?
2์ผ์ฐจ ๊ฐ์
์๋นํ ๋ถ๋์ ๋ฌ์คํธ์ ๋ํด ๋ณด์๊ณ , ์ด์ด์ ์ค๋ ๊ฐ์๋ฅผ ์งํํ๊ฒ ์ต๋๋ค:
-
๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ: ์คํ๊ณผ ํ, ์๋ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ, ์ค์ฝํ(๋ฒ์)๊ธฐ๋ฐ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ, ๊ฐ๋น์ง ์ปฌ๋ ์ (GC)
-
์์ ๊ถ: Move ๋ฌธ๋ฒ, ๋ณต์ฌ์ ๋ณต์ , ๋น๋ฆผ, ์๋ช .
-
Structs and methods.
-
ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ:
String
,Option
๊ณผResult
,Vec
,HashMap
,Rc
๊ทธ๋ฆฌ๊ณArc
. -
๋ชจ๋: ๊ฐ์์ฑ, ๊ฒฝ๋ก ๋ฐ ํ์ผ ์์คํ ๊ณ์ธต.
๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ
์ ํต์ ์ผ๋ก, ๋ ์ข ๋ฅ์ ํ๋ก๊ทธ๋๋ฐ ์ธ์ด๊ฐ ์์ต๋๋ค:
- ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ๊ฐ ํ๋ก๊ทธ๋๋จธ์ ์์ ํ ํต์ ํ์ ์์ง๋ง ์๋(๊ทธ๋์ ์์ ํ์ง ์์ ์ ์๋)์ธ ์ธ์ด: C, C++, Pascal, โฆ
- ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ๊ฐ ๋ฐํ์์ ์ํด ๋๋ฏ๋ก ์์ ํ์ง๋ง ์๋(๊ทธ๋์ ํ๋ก๊ทธ๋๋จธ๊ฐ ๊ฐ์ ํ ์ฌ์ง๊ฐ ์ ๊ฑฐ๋ ์๋)์ธ ์ธ์ด: Java, Python, Go, Haskell, โฆ
๋ฌ์คํธ๋ ์ด ๋์ ํผํฉํ ์๋ก์ด ํํ์ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ ๊ธฐ๋ฒ์ ์ ๊ณตํฉ๋๋ค:
์ปดํ์ผ ์ ์ฌ๋ฐ๋ฅธ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ๋ฅผ ๊ฐ์ ํจ์ผ๋ก์จ ์์ ํ ํต์ ์ ์์ ์ฑ ๋ชจ๋ ์ ๊ณต.
์ด๋ฅผ ๊ฐ๋ฅํ๊ฒ ํ๋ ๋ฌ์คํธ์ ์ปจ์ ์ ๋ช ์์ ์ธ ์์ ๊ถ์ ๋๋ค.
์ฐ์ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ๊ฐ ์ด๋ค์ง๋ ๋ฐฉ์์ ๋ค์ ์ดํด ๋ณด๊ฒ ์ต๋๋ค.
์คํ(Stack)๊ณผ ํ(Heap)
-
์คํ: ๋ก์ปฌ ๋ณ์๋ฅผ ์ํ ์ฐ์์ ์ธ ๋ฉ๋ชจ๋ฆฌ ์์ญ.
- ์ฌ๊ธฐ ์ ์ฅ๋๋ ๊ฐ์ ์ปดํ์ผ ์ ๊ฒฐ์ ๋๋ ๊ณ ์ ํฌ๊ธฐ๋ฅผ ๊ฐ์ต๋๋ค.
- ๋งค์ฐ ๋น ๋ฆ: ๋ฉ๋ชจ๋ฆฌ ํ ๋น/๋ฐํ์ด ๋จ์ง ์คํ ํฌ์ธํฐ์ ์ด๋๋ง์ผ๋ก ๊ตฌํ๋ฉ๋๋ค.
- ๊ด๋ฆฌ๊ฐ ์ฌ์: ํจ์๊ฐ ํธ์ถ๋๋ฉด ํ ๋น๋๊ณ , ๋ฆฌํดํ๋ฉด ๋ฐํ๋ฉ๋๋ค.
- ์คํ์ ์๋ ๊ฐ๋ค์ ๋งค์ฐ ๋์ ๋ฉ๋ชจ๋ฆฌ ์ธ์ ์ฑ์ ๊ฐ์ง๋๋ค.
-
ํ: ํจ์ ํธ์ถ/๋ฆฌํด๊ณผ ์๊ด ์์ด ์ ์ง๋๋ ๊ฐ์ด ์ ์ฅ๋๋ ๊ณณ.
- ์ฌ๊ธฐ ์ ์ฅ๋๋ ๊ฐ์ ํ๋ก๊ทธ๋จ ์ํ์ ๊ทธ ํฌ๊ธฐ๊ฐ ๊ฒฐ์ ๋ฉ๋๋ค.
- ์คํ ๋ณด๋ค๋ ๋๋ฆผ: ๋ฉ๋ชจ๋ฆฌ ํ ๋น/๋ฐํ์ ํด์ผ ํ ์ผ์ด ์ข ๋ ์์ต๋๋ค.
- ๋ฉ๋ชจ๋ฆฌ ์ธ์ ์ฑ์ ๋ณด์ฅํ์ง ์์ต๋๋ค.
์คํ๊ณผ ํ์ ๊ดํ ์์
String
์ ํ๋ ๋ง๋ค๊ฒ ๋๋ฉด, ์คํ์๋ ๊ณ ์ ๋ ํฌ๊ธฐ์ ๋ฉํ ๋ฐ์ดํฐ๊ฐ ์์ฑ๋๊ณ , ํ์๋ ๊ฐ๋ณ ํฌ๊ธฐ์ ๋ฐ์ดํฐ, ์ฆ, ์ค์ ๋ฌธ์์ด, ์ด ์์ฑ๋ฉ๋๋ค:
fn main() { let s1 = String::from("Hello"); }
-
๋ฌธ์์ด(
String
)์ ์ค์ ๋ก๋Vec
์ ๋๋ค. ํฌ๊ธฐ(capacity)์ ํ์ฌ ๊ธธ์ด(length) ์ ๋ณด๋ฅผ ๊ฐ์ง๋ฉฐ, ๋ ํฐ ํฌ๊ธฐ๊ฐ ํ์ํ ๊ฒฝ์ฐ ํ์์ ์ฌ ํ ๋น์ ํฉ๋๋ค. -
ํ์ ๊ธฐ๋ณธ์ ์ผ๋ก System Allocator๋ฅผ ํตํด ํ ๋น๋ฉ๋๋ค. ๊ทธ๋ฆฌ๊ณ Allocator API๋ฅผ ์ด์ฉํด์ ์ปค์คํ ๋ฉ๋ชจ๋ฆฌ ํ ๋น์๋ฅผ ๋ง๋ค ์๋ ์์ต๋๋ค.
-
์๋์ ๊ฐ์
unsafe
์ฝ๋๋ก ๋ฉ๋ชจ๋ฆฌ ๋ ์ด์์์ ์ดํด๋ณผ ์ ์์ต๋๋ค. ๋ฌผ๋ก ์ด ์ฝ๋๊ฐ ์์ ํ์ง ์๋ค๋ ์ ์ ์๋ ค์ฃผ์ธ์!fn main() { let mut s1 = String::from("Hello"); s1.push(' '); s1.push_str("world"); // DON'T DO THIS AT HOME! For educational purposes only. // String provides no guarantees about its layout, so this could lead to // undefined behavior. unsafe { let (ptr, capacity, len): (usize, usize, usize) = std::mem::transmute(s1); println!("ptr = {ptr:#x}, len = {len}, capacity = {capacity}"); } }
์๋ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ
์ฌ์ฉ์๊ฐ ์ง์ ๋ฉ๋ชจ๋ฆฌ๋ฅผ ํ ๋น, ํด์ ํฉ๋๋ค.
์กฐ์ฌํ์ง ์์ผ๋ฉด, ์ถฉ๋(crash), ๋ฒ๊ทธ, ๋ณด์์ทจ์ฝ์ฑ ๋ฐ ๋ฉ๋ชจ๋ฆฌ ๋์ถ์ด ๋ฐ์ํ ์ ์์ต๋๋ค.
C ์ธ์ด ์์
malloc
์ผ๋ก ํ ๋นํ๋ ํฌ์ธํฐ๋ง๋ค free
๋ฅผ ํธ์ถํด์ผ ํฉ๋๋ค:
void foo(size_t n) {
int* int_array = malloc(n * sizeof(int));
//
// ... lots of code
//
free(int_array);
}
๋ง์ฝ malloc
๊ณผ free
์ฌ์ด์์ ํจ์๊ฐ ์ผ์ฐ ๋ฐํ๋๋ฉด ๋ฉ๋ชจ๋ฆฌ ๋์ถ์ด ์ผ์ด๋ฉ๋๋ค: ํฌ์ธํฐ๋ฅผ ์์ด๋ฒ๋ฆฌ๊ฒ ๋์ด ๋ฉ๋ชจ๋ฆฌ๋ฅผ ๋ฐํํ ์ ์๊ฒ ๋ฉ๋๋ค. ๊ฐ์ ํฌ์ธํฐ๋ฅผ ๋ ๋ฒ ๋ฐํํ๊ฑฐ๋, ์ด๋ฏธ ๋ฐํ๋ ํฌ์ธํฐ๋ฅผ ์ ๊ทผํ๋ ๊ฒ์ ์ฌ๊ฐํ ๋ณด์ ๋ฌธ์ ๋ฅผ ์ผ์ผํฌ ์ ์์ต๋๋ค.
๋ฒ์๊ธฐ๋ฐ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ
์์ฑ์์ ์๋ฉธ์๋ฅผ ์ฌ์ฉํ์ฌ ๊ฐ์ฒด์ ์๋ช ์ฃผ๊ธฐ์ ๋ฐ๋ผ ๋ฉ๋ชจ๋ฆฌ ํ ๋น/ํด์ ๊ฐ ์ผ์ด๋๋๋ก ํ ์ ์์ต๋๋ค.
ํฌ์ธํฐ๋ฅผ ๊ฐ์ฒด๋ก ๊ฐ์ธ๋๋ก ํ๋ฉด, ๊ทธ ๊ฐ์ฒด๊ฐ ์๋ฉธ๋ ๋ ๊ทธ ํฌ์ธํฐ๊ฐ ๊ฐ๋ฆฌํค๋ ๋ฉ๋ชจ๋ฆฌ๊ฐ ํด์ ๋๋๋ก ํ ์ ์์ต๋๋ค. ์ปดํ์ผ๋ฌ๋ ๊ฐ์ฒด๊ฐ ์๋ฉธ๋ ๋ ๋ฐ๋์ ์๋ฉธ์๊ฐ ํธ์ถ๋๋ ๊ฒ์ ๋ณด์ฅํฉ๋๋ค. ์ฌ์ง์ด๋ ์์ธ(exception)๊ฐ ๋ฐ์(์ญ์ฃผ: ํจ์์ ๋ฆฌํด์ด๋ ์ค์ฝํ์ ์ข ๋ฃ ๋ฟ๋ง์ด ์๋๋ผ) ํ๋๋ผ๋์.
์ด๋ฅผ ์ข ์ข RAII (Resource Acquisition Is Initialization)๋ผ๊ณ ํ๋ฉฐ, ์ด๋ฐ ๊ฐ์ฒด๋ ์ผ์ข ์ ์ค๋งํธ ํฌ์ธํฐ ์ญํ ์ ํฉ๋๋ค.
C++ ์์
void say_hello(std::unique_ptr<Person> person) {
std::cout << "Hello " << person->name << std::endl;
}
std::unique_ptr
๊ฐ์ฒด๋ ์คํ์ ํ ๋น๋๋ฉฐ, ํ์ ํ ๋น๋ ๋ฉ๋ชจ๋ฆฌ๋ฅผ ๊ฐ๋ฆฌํต๋๋ค.say_hello
ํจ์๊ฐ ๋๋๋ฉดstd::unique_ptr
์ ์๋ฉธ์๊ฐ ์คํ๋ฉ๋๋ค.- ์๋ฉธ์๋
Person
๊ฐ์ฒด๊ฐ ๊ฐ๋ฆฌํค๋ ๋ฉ๋ชจ๋ฆฌ๋ฅผ ํด์ ํฉ๋๋ค.
์ด๋ ์์ฑ์๋ ํจ์ ํธ์ถ ์ ์์ ๊ถ์ ์ ๋ฌํ ๋ ์ฌ์ฉ๋ฉ๋๋ค:
std::unique_ptr<Person> person = find_person("Carla");
say_hello(std::move(person));
์๋ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ
์๋, ์ค์ฝํ๊ธฐ๋ฐ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ์ ๋์์ผ๋ก ์๋ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ ๋ฐฉ์์ด ์์ต๋๋ค:
- ๊ฐ๋ฐ์๋ ๋ฉ๋ชจ๋ฆฌ๋ฅผ ๋ช ์์ ์ผ๋ก ํ ๋น/ํด์ ํ์ง ์์ต๋๋ค.
- ๊ฐ๋น์ง ์ปฌ๋ ํฐ(GC)๋ ์ฌ์ฉ๋์ง ์๋ ๋ฉ๋ชจ๋ฆฌ๋ฅผ ์ฐพ์ ํด์ ํฉ๋๋ค.
Java ์์
person
๊ฐ์ฒด๋ sayHello
ํจ์ ๋ฐํ ํ์๋ ํด์ ๋์ง ์์ต๋๋ค. (์ญ์ฃผ: GC๊ฐ ๋์ค์ ์์์ ํด์ ํฉ๋๋ค.)
void sayHello(Person person) {
System.out.println("Hello " + person.getName());
}
๋ฌ์คํธ์์์ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ
๋ฌ์คํธ์ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ๋ ์ง๊ธ๊น์ง ์ค๋ช ํ ๋ฐฉ์๋ค์ ํผํฉํด์ ์ฌ์ฉํฉ๋๋ค:
- ์๋ฐ์ฒ๋ผ ์์ ํ๊ณ ์ ํํฉ๋๋ค. ํ์ง๋ง GC๋ ์์ต๋๋ค.
- C++ ์ฒ๋ผ ๋ฒ์(์ค์ฝํ) ๊ธฐ๋ฐ์ ๋๋ค. ํ์ง๋ง ์ปดํ์ผ๋ฌ๊ฐ ํจ์ฌ ๋ ์๊ฒฉํฉ๋๋ค.
- ์ฌ์ฉ์๋ ์ํฉ์ ๋ฐ๋ผ ์ ํฉํ ์ถ์ํ๋ฅผ ์ ํํ ์ ์์ต๋๋ค. ๊ทธ ์ค์๋ C ์ธ์ด ์ฒ๋ผ ๋ฐํ์ ์ค๋ฒํค๋๊ฐ ์๋ ๊ฒ๋ ์์ต๋๋ค.
๋ฌ์คํธ๋ _์์ ๊ถ_์ ์ธ์ด ์ฐจ์์์ ๋ช ์์ ์ผ๋ก ๋ชจ๋ธ๋ง ํจ์ผ๋ก์จ ์ด๋ฅผ ์ด๋ฃน๋๋ค.
-
์ด ์์ ์์ ๊ทธ๊ฒ ์ด๋ป๊ฒ ๊ฐ๋ฅํ๋๋ ์ง๋ฌธ์ด ์์ผ๋ฉด, ๋ฌ์คํธ์์ ์ด ์์ ์ ์ผ๋ฐ์ ์ผ๋ก Box, Vec, Rc ๋๋ Arc์ ๊ฐ์ RAII ํ์ ์ ์ํด ์ฒ๋ฆฌ๋๋ค๊ณ ๋ต๋ณํ ์ ์์ต๋๋ค. ์ด๋ค์ ๋ค์ํ ๋ฐฉ๋ฒ์ ํตํด ์์ ๊ถ๊ณผ ๋ฉ๋ชจ๋ฆฌ ํ ๋น์ ๋ํ ๊ตฌ์ฒด์ ์ธ ๋ด์ฉ์์ ์บก์ํํ์ฌ, C ์ธ์ด์๋ค๋ฉด ๋ฐ์ํ ์ ์์์ ๋ค์ํ ์๋ฌ๋ฅผ ๋ง์ต๋๋ค.
-
์๋ฉธ์์ ๋ํ ์ง๋ฌธ๋ ์์ ์ ์์ต๋๋ค. Drop ํธ๋ ์์ด ๋ต์ ๋๋ค.
์์ ๊ถ
๋ชจ๋ ๋ณ์ ๋ฐ์ธ๋ฉ์ ์ ํจํ โ๋ฒ์(์ค์ฝํ)โ๋ฅผ ๊ฐ์ง๋ฉฐ, ๋ฒ์ ๋ฐ์์ ๋ณ์ ์ฌ์ฉํ๋ฉด ์๋ฌ๊ฐ ๋ฐ์ํฉ๋๋ค:
struct Point(i32, i32); fn main() { { let p = Point(3, 4); println!("x: {}", p.0); } println!("y: {}", p.1); }
- ์ค์ฝํ๊ฐ ์ข ๋ฃ๋๋ฉด ๋ณ์๋ โ์ญ์ (drop)โ๋์๋ค๊ณ ํ๋ฉฐ ๊ทธ ๋ณ์์ ๋ฐ์ดํฐ๋ ๋ฉ๋ชจ๋ฆฌ์์ ํด์ ๋ฉ๋๋ค.
- ์ค์ฝํ๊ฐ ์ข ๋ฃ๋ ๋ ๋ค๋ฅธ ๋ฆฌ์์ค๋ฅผ ํด์ ํ๊ธฐ ์ํด ์๋ฉธ์๊ฐ ํธ์ถ๋๋๋ก ํ ์ ์์ต๋๋ค.
- ์ด๊ฒ์ ๋๊ณ ๋ณ์๊ฐ ๊ฐ์ โ์์ โํ๋ค๊ณ ํํํฉ๋๋ค.
Move ๋ฌธ๋ฒ
(๋ณ์์) ํ ๋น์ _์์ ๊ถ_์ ๋ณ์ ๊ฐ์ ์ด๋์ํต๋๋ค:
fn main() { let s1: String = String::from("Hello!"); let s2: String = s1; println!("s2: {s2}"); // println!("s1: {s1}"); }
s1
์s2
์ ํ ๋นํ์ฌ ์์ ๊ถ์ ์ด์ ์ํต๋๋ค.s1
์ ์ค์ฝํ๊ฐ ์ข ๋ฃ๋๋ฉด ์๋ฌด ์ผ๋ ์์ต๋๋ค: ์๋ํ๋ฉดs1
์ ์๋ฌด๋ฐ ์์ ๊ถ์ด ์๊ธฐ ๋๋ฌธ์ ๋๋ค.s2
์ ์ค์ฝํ๊ฐ ์ข ๋ฃ๋๋ฉด ๋ฌธ์์ด ๋ฐ์ดํฐ๋ ํด์ ๋ฉ๋๋ค.- ๊ฐ(๋ฐ์ดํฐ)์ ์์ ๊ถ์ ๊ฐ๋ ๋ณ์๋ ํญ์ ๋จ ํ๋ ์ ๋๋ค.
-
์ด๋ C++๊ณผ ์ ๋ฐ๋ ์์ ์ค๋ช ํ์ธ์. C++์์๋ ๋ณต์ฌ๊ฐ ๊ธฐ๋ณธ์ด๊ณ ,
std::move
๋ฅผ ์ด์ฉํด์ผ๋ง (๊ทธ๋ฆฌ๊ณ ์ด๋ ์์ฑ์๊ฐ ์ ์๋์ด ์์ด์ผ๋ง!) ์์ ๊ถ ์ด์ ์ด ๋ฉ๋๋ค. -
์ค์ ๋ก ์ด๋๋๋ ๊ฒ์ ์์ ๊ถ์ผ ๋ฟ์ ๋๋ค. ๋จธ์ ์ฝ๋ ๋ ๋ฒจ์์ ๋ฐ์ดํฐ ๋ณต์ฌ๊ฐ ์ผ์ด๋ ์ง ๋ง ์ง์ ๋ํ ๊ฒ์ ์ปดํ์ผ๋ฌ ๋ด๋ถ์์ ์ผ์ด๋๋ ์ต์ ํ ๋ฌธ์ ์ ๋๋ค. ์ด๋ฐ ๋ณต์ฌ๋ ์ต์ ํ ๊ณผ์ ์์ ์ ๊ฑฐ๊ฐ ๋ฉ๋๋ค.
-
์ ์์ ๊ฐ์ ๊ฐ๋จํ ๊ฐ๋ค์
Copy
(๋ค์ ์ค๋ช ํฉ๋๋ค)๋ก ๋งํน๋ ์ ์์ต๋๋ค. -
๋ฌ์คํธ์์๋ ๋ณต์ฌํ ๋์๋ ๋ช ์์ ์ผ๋ก
clone
์ ์ฌ์ฉํฉ๋๋ค.
๋ฌ์คํธ์์์ ๋ฌธ์์ด ์ด๋
fn main() { let s1: String = String::from("Rust"); let s2: String = s1; }
s1
์ ํ ๋ฐ์ดํฐ๋s2
์์ ์ฌ์ฌ์ฉ ๋ฉ๋๋ค.s1
์ ์ค์ฝํ๊ฐ ์ข ๋ฃ๋๋ฉด ์๋ฌด์ผ๋ ์ผ์ด๋์ง ์์ต๋๋ค.(์ด๋ฏธ ์ด๋๋์์ต๋๋ค.)
s2
๋ก ์ด๋ ์ ๋ฉ๋ชจ๋ฆฌ:
s2
๋ก ์ด๋ ํ ๋ฉ๋ชจ๋ฆฌ:
Defensive Copies in Modern C++
Modern C++์ ์ด ๋ฌธ์ ๋ฅผ ๋ค๋ฅด๊ฒ ํด๊ฒฐํฉ๋๋ค:
std::string s1 = "Cpp";
std::string s2 = s1; // Duplicate the data in s1.
s1
์ ํ ๋ฐ์ดํฐ๋ ๋ณต์ ๋๊ณ ,s2
๋ ๋ ๋ฆฝ์ ์ธ ๋ณต์ฌ๋ณธ์ ์ป์ต๋๋ค.s1
์s2
์ ์ค์ฝํ๊ฐ ์ข ๋ฃ๋๋ฉด ๊ฐ๊ฐ์ ๋ฉ๋ชจ๋ฆฌ๊ฐ ํด์ ๋ฉ๋๋ค.
๋ณต์ฌ ์ :
๋ณต์ฌ ํ:
ํค ํฌ์ธํธ:
-
C++ has made a slightly different choice than Rust. Because
=
copies data, the string data has to be cloned. Otherwise we would get a double-free when either string goes out of scope. -
C++ also has
std::move
, which is used to indicate when a value may be moved from. If the example had beens2 = std::move(s1)
, no heap allocation would take place. After the move,s1
would be in a valid but unspecified state. Unlike Rust, the programmer is allowed to keep usings1
. -
Unlike Rust,
=
in C++ can run arbitrary code as determined by the type which is being copied or moved.
ํจ์ ํธ์ถ์์์ ์ด๋(Move)
๊ฐ์ ํจ์์ ์ ๋ฌํ ๋, ๊ทธ ๊ฐ์ ๋งค๊ฐ๋ณ์์ ํ ๋น๋ฉ๋๋ค. ์ด๋ ์์ ๊ถ์ ์ด๋์ด ์ผ์ด๋ฉ๋๋ค:
fn say_hello(name: String) { println!("Hello {name}") } fn main() { let name = String::from("Alice"); say_hello(name); // say_hello(name); }
say_hello
ํจ์์ ์ฒซ๋ฒ์งธ ํธ์ถ์main
ํจ์๋ ์์ ์ด ๊ฐ์งname
์ ๋ํ ์์ ๊ถ์ ํฌ๊ธฐํ๋ฏ๋ก, ์ดํmain
ํจ์์์๋name
์ ์ฌ์ฉํ ์ ์์ต๋๋ค.name
์ ํ ๋น๋์๋ ํ ๋ฉ๋ชจ๋ฆฌ๋say_hello
ํจ์์ ๋์์ ํด์ ๋ฉ๋๋ค.main
ํจ์์์name
์ ์ฐธ์กฐ๋ก ์ ๋ฌ(๋น๋ฆผ)ํ๊ณ (&name
),say_hello
์์ ๋งค๊ฐ๋ณ์๋ฅผ ์ฐธ์กฐํ์ผ๋ก ์์ ํ๋ค๋ฉดmain
ํจ์๋name
์ ์์ ๊ถ์ ์ ์งํ ์ ์์ต๋๋ค.- ๋๋ ์ฒซ๋ฒ์งธ ํธ์ถ ์
main
ํจ์์์name
์ ๋ณต์ ํ์ฌ ์ ๋ฌํ ์๋ ์์ต๋๋ค.(name.clone()
) - ๋ฌ์คํธ๋ ์ด๋์ ๊ธฐ๋ณธ์ผ๋ก ํ๊ณ ๋ณต์ ๋ฅผ ๋ช ์์ ์ผ๋ก ์ ์ธํ๋๋ก ๋ง๋ฌ์ผ๋ก, ์๋์น ์๊ฒ ๋ณต์ฌ๋ณธ์ ๋ง๋๋ ๊ฒ์ด C++์์๋ณด๋ค ์ด๋ ต์ต๋๋ค.
๋ณต์ฌ์ ๋ณต์
์ด๋์ด ๊ธฐ๋ณธ ์ค์ ์ด์ง๋ง, ํน์ ํ์ ์ ๋ณต์ฌ๋ฉ๋๋ค:
fn main() { let x = 42; let y = x; println!("x: {x}"); println!("y: {y}"); }
์ด๋ฌํ ํ์
๋ค์ Copy
ํธ๋ ์์ ๊ตฌํํฉ๋๋ค.
์ง์ ๋ง๋ ํ์
๋ค๋ Copy
ํธ๋ ์์ ๊ตฌํํ์ฌ ๋ณต์ฌ๋ฅผ ํ ์ ์์ต๋๋ค:
#[derive(Copy, Clone, Debug)] struct Point(i32, i32); fn main() { let p1 = Point(3, 4); let p2 = p1; println!("p1: {p1:?}"); println!("p2: {p2:?}"); }
- ํ ๋น ํ,
p1
์p2
๋ ์์ ์ ๋ฐ์ดํฐ๋ฅผ ์์ ํฉ๋๋ค. - ๋ช
์์ ์ผ๋ก
p1.clone()
๋ฅผ ์ฌ์ฉํ์ฌ ๋ฐ์ดํฐ๋ฅผ ๋ณต์ฌํ ์ ์์ต๋๋ค.
๋ณต์ฌ(copy)์ ๋ณต์ (clone)๋ ๊ฐ์ง ์์ต๋๋ค:
- ๋ณต์ฌ๋ ๋ฉ๋ชจ๋ฆฌ์ ๋ด์ฉ์ ๊ทธ๋๋ก ํ ๋ฒ ๋ ๋ง๋๋ ๊ฒ์ ์๋ฏธํ๋ฉฐ, ์๋ฌด ๊ฐ์ฒด์์๋ ๋ค ์ง์ํ์ง๋ ์์ต๋๋ค.
- ๋ณต์ฌ๋ ์ปค์คํฐ๋ง์ด์ฆ ํ ์ ์์ต๋๋ค. (C++์์ ๋ณต์ฌ ์์ฑ์๋ฅผ ํตํด ๋ณต์ฌ ๋์์ ์์๋ก ๊ตฌํํ ์ ์๋ ๊ฒ๊ณผ ๋น๊ต๊ฐ ๋ฉ๋๋ค.)
- ๋ณต์ ๋ ๋ณด๋ค ์ผ๋ฐ์ ์ธ ์์
์ด๋ฉฐ,
Clone
ํธ๋ ์์ ๊ตฌํํ์ฌ ๋ณต์ ์ ๋์์ ์ปค์คํฐ๋ง์ด์ฆ ํ ์ ์์ต๋๋ค. Drop
ํธ๋ ์์ ๊ตฌํํ ํ์ ์ ๋ณต์ฌ๋์ง ์์ต๋๋ค.
์์ ์์์์ ๋ค์์ ์๋ํด ๋ณด์๊ธฐ ๋ฐ๋๋๋ค:
Point
๊ตฌ์กฐ์ฒด์String
ํ๋๋ฅผ ์ถ๊ฐํ์ธ์. ์ปดํ์ผ ๋์ง ์์ ๊ฒ์ ๋๋ค. ์๋ํ๋ฉดString
์Copy
ํธ๋ ์์ ๊ตฌํํ๊ณ ์์ง ์๊ธฐ ๋๋ฌธ์ ๋๋ค.derive
์์ฑ์์Copy
๋ฅผ ์ ๊ฑฐํด ๋ณด์ธ์.p1
์println!
ํ ๋ ์ปดํ์ผ ์๋ฌ๊ฐ ๋ฐ์ํ ๊ฒ์ ๋๋ค.p1
์ ๋ณต์ ํ๋ฉด ์ ๋์ํจ์ ํ์ธํด ๋ณด์ธ์.
๋ง์ฝ ํ์๋ค์ด derive
์ ๋ํด ๋ฌป๋๋ค๋ฉด, ์ปดํ์ผ ์ ๋ฌ์คํธ์์ ์ฝ๋๋ฅผ ์์ฑํ๋๋ฐฉ๋ฒ์ด๋ผ๊ณ ๋งํ๋ ๊ฒ์ผ๋ก ์ถฉ๋ถํฉ๋๋ค. ์ ๊ฒฝ์ฐ Copy
์ Clone
ํธ๋ ์์ ๋ํ ๊ธฐ๋ณธ ๊ตฌํ์ด ์์ฑ๋ฉ๋๋ค.
๋น๋ฆผ
ํจ์ ํธ์ถ์ ๊ฐ์ ์์ ๊ถ์ ์ด๋ํ๋ ๋์ ์ ํจ์๊ฐ ๊ฐ์ ๋น๋ ค์ฌ ์ ์์ต๋๋ค:
#[derive(Debug)] struct Point(i32, i32); fn add(p1: &Point, p2: &Point) -> Point { Point(p1.0 + p2.0, p1.1 + p2.1) } fn main() { let p1 = Point(3, 4); let p2 = Point(10, 20); let p3 = add(&p1, &p2); println!("{p1:?} + {p2:?} = {p3:?}"); }
add
ํจ์๋ ๋Point
๊ฐ์ฒด ๊ฐ์ _๋น๋ ค_์์ ์๋ก์ดPoint
๊ฐ์ฒด๋ฅผ ๋ฐํํฉ๋๋ค.p1
๊ณผp2
์ ์์ ๊ถ์ ์ฌ์ ํ ํธ์ถ์(main
ํจ์)์ ์์ต๋๋ค.
์คํ์ ํ ๋น๋ ๊ฐ์ ๋ฆฌํดํ๋ ๊ฒ์ ๋ํ ์ฐธ๊ณ :
-
add
์์ ๊ฐ์ ๋ฐํํ๋ ๊ฒ์ ๋งค์ฐ ๊ฐ์ด ์ธ๋ค๋ ๊ฒ์ ์ค๋ช ํ์ธ์. ์๋ํ๋ฉด, ์ปดํ์ผ๋ฌ๊ฐ ๋ณต์ฌ ๊ณผ์ ์ ์๋ตํ ์ ์๊ธฐ ๋๋ฌธ์ ๋๋ค. ์ ์ฝ๋๋ฅผ ์คํ ์ฃผ์๋ฅผ ์ถ๋ ฅํ๋๋ก ์์ ํ๊ณ Playground์์ ์ํํด ๋ณด์ธ์. ๋๋ Godbolt์์ ์ด์ ๋ธ๋ฆฌ๋ฅผ ํ์ธํด ๋ณด์ธ์. ์ต์ ํ ๋ ๋ฒจ์ด โDEBUGโ ์ผ ๋์๋ ์ฃผ์๊ฐ ๋ฐ๋์ง๋ง, โRELEASEโ ๋ ๋ฒจ์์๋ ๋ฐ๋์ง ์์ต๋๋ค:#[derive(Debug)] struct Point(i32, i32); fn add(p1: &Point, p2: &Point) -> Point { let p = Point(p1.0 + p2.0, p1.1 + p2.1); println!("&p.0: {:p}", &p.0); p } pub fn main() { let p1 = Point(3, 4); let p2 = Point(10, 20); let p3 = add(&p1, &p2); println!("&p3.0: {:p}", &p3.0); println!("{p1:?} + {p2:?} = {p3:?}"); }
-
๋ฌ์คํธ ์ปดํ์ผ๋ฌ๋ ๋ฐํ๊ฐ ์ต์ ํ(RVO)๋ฅผ ์ํํ ์ ์์ต๋๋ค.
-
C++์์ copy elision์ ์์ฑ์์ ๋ถ์ํจ๊ณผ ๊ฐ๋ฅ์ฑ์ด ์์ด ์ธ์ด๋ ๋ฒจ์ ์ ์๊ฐ ํ์ํ์ง๋ง ๋ฌ์คํธ์์๋ ๋ฌธ์ ๊ฐ ๋์ง ์์ต๋๋ค. ๋ง์ฝ RVO๊ฐ ๋ฐ์ํ์ง ์์ผ๋ฉด ๋ฌ์คํธ๋ ํญ์ ๊ฐ๋จํ๊ณ ํจ์จ์ ์ธ
memcpy
๋ณต์ฌ๋ฅผ ์ํํ ๊ฒ์ ๋๋ค.
๊ณต์ ์ ๊ณ ์ ๋น๋ฆผ
๋ฌ์คํธ์์๋ ๊ฐ์ ๋น๋ฆด ๋ ๋ค์๊ณผ ๊ฐ์ ์ ์ฝ์กฐ๊ฑด์ด ์์ต๋๋ค:
- ํ๋ฒ์ ํ๋ ์ด์์
&T
๊ฐ์ ๊ฐ์ง๊ฑฐ๋, ๋๋ - ์ ํํ ํ๋์
&mut T
๊ฐ๋ง์ ๊ฐ์ง ์ ์์ต๋๋ค.
fn main() { let mut a: i32 = 10; let b: &i32 = &a; { let c: &mut i32 = &mut a; *c = 20; } println!("a: {a}"); println!("b: {b}"); }
- ์ ์ฝ๋ ์ปดํ์ผ ๋์ง ์์ต๋๋ค. ์๋ํ๋ฉด
c
๋a
๋ฅผ ๊ฐ๋ณ ๋ณ์๋ก ๋น๋ ธ๊ณ , ์ด์ ๋์์b
๋a
๋ฅผ ๋ถ๋ณ ๋ณ์๋ก ๋น๋ ธ๊ธฐ ๋๋ฌธ์ ๋๋ค. b
์ ๋ํprintln!
๊ตฌ๋ถ์c
๊ฐ ์๋ ์ค์ฝํ ์์ผ๋ก ์ด๋ํ๋ฉด ์ปดํ์ผ์ด ๋ฉ๋๋ค.- ์ด๋ ๊ฒ ๋ฐ๊พธ๋ฉด, ์ปดํ์ผ๋ฌ๋
c
๊ฐa
๋ฅผ ๊ฐ๋ณ ๋ณ์๋ก ๋น๋ฆฌ๊ธฐ ์ ์๋งb
๊ฐ ์ฌ์ฉ๋๋ค๋ ๊ฒ์ ํ์ธํ ์ ์์ต๋๋ค. ๋น๋ฆผ ๊ฒ์ฌ๊ธฐ์ ์ด๋ฌํ ๊ธฐ๋ฅ์ โnon-lexical lifetimeโ ์ด๋ผ๊ณ ํฉ๋๋ค.
์๋ช
๋น๋ ค์จ ๊ฐ์ _์๋ช _์ ๊ฐ์ต๋๋ค:
- ์๋ช
์ ์๋ตํ ์ ์์ต๋๋ค:
add(p1: &Point, p2: &Point) -> Point
. - ๋ฌผ๋ก ๋ช
์ํ ์๋ ์์ต๋๋ค:
&'a Point
,&'document str
. &'a Point
๋Point
์ ์๋ช ์ด ์ต์ํ'a
๋ผ๋ ์๋ช ๋ณด๋ค๋ ๊ฐ๊ฑฐ๋ ๋ ๊ธธ๋ค๋ ๊ฒ์ ์๋ฏธํฉ๋๋ค.- ์๋ช
์ ํญ์ ์ปดํ์ผ๋ฌ๊ฐ ์๋์ผ๋ก ์ถ๋ก ํฉ๋๋ค. ์ง์ ์๋ช
์ ์ง์ ํ ์๋ ์์ต๋๋ค.
- ์๋ช
ํ๊ธฐ(
'
)์ ์๋ช ์ถ๋ก ์ ์ ์ฝ์กฐ๊ฑด์ด ๋ฉ๋๋ค. ์ปดํ์ผ๋ฌ๋ ์ด ์ ์ฝ์กฐ๊ฑด์ ๋ง์กฑ์ํค๋ ์ ์ํ ์๋ช ์ ์ถ๋ก ํ ์ ์๋์ง ๊ฒ์ฌ๋ฅผ ํฉ๋๋ค.
- ์๋ช
ํ๊ธฐ(
- ์๋, ํจ์์ ์ธ์์ ๋ฆฌํด๊ฐ์๋ ์๋ช ์ ์ค์ ํด์ผ ํฉ๋๋ค. ๊ทธ๋ฌ๋ ๋ช ๊ฐ์ง ์กฐ๊ฑด์ ๋ง์กฑํ๋ ๋๋ถ๋ถ์ ๊ฒฝ์ฐ์๋ ์๋ตํ ์ ์์ต๋๋ค.
ํจ์ ํธ์ถ์์์ ์๋ช
ํจ์๋ ์ธ์๋ฅผ ๋น๋ฆฌ๋ ๊ฒ ์ธ์๋ ๋น๋ฆฐ ๊ฐ์ ๋ฐํํ ์ ์์ต๋๋ค:
#[derive(Debug)] struct Point(i32, i32); fn left_most<'a>(p1: &'a Point, p2: &'a Point) -> &'a Point { if p1.0 < p2.0 { p1 } else { p2 } } fn main() { let p1: Point = Point(10, 10); let p2: Point = Point(20, 20); let p3: &Point = left_most(&p1, &p2); println!("left-most point: {:?}", p3); }
'a
๋ ์ ๋ค๋ฆญ ๋งค๊ฐ๋ณ์๋ก ์ปดํ์ผ๋ฌ๋ก์ ์ํด ์ถ๋ก ๋ฉ๋๋ค.- ์๋ช
์ ์ด๋ฆ์
'
๋ก ์์ํ๋ฉฐ ๋ณดํต'a
๋ฅผ ๋ง์ด ์๋๋ค. &'a Point
๋Point
์ ์๋ช ์ด ์ต์ํ'a
๋ผ๋ ์๋ช ๋ณด๋ค๋ ๊ฐ๊ฑฐ๋ ๋ ๊ธธ๋ค๋ ๊ฒ์ ์๋ฏธํฉ๋๋ค.- ๋งค๊ฐ๋ณ์๋ค์ด ์๋ก ๋ค๋ฅธ ์ค์ฝํ์ ์์ ๊ฒฝ์ฐ โ์ต์ํโ์ด๋ผ๋ ์กฐ๊ฑด์ด ์ค์ํฉ๋๋ค.
์์ ์์์์ ๋ค์์ ์๋ํด ๋ณด์๊ธฐ ๋ฐ๋๋๋ค:
-
p2
์p3
๋ฅผ ์๋ก์ด ๋ฒ์({...}
)๋ก ์๋ ์ฝ๋์ ๊ฐ์ด ์ด๋ํด ๋ด ๋๋ค:#[derive(Debug)] struct Point(i32, i32); fn left_most<'a>(p1: &'a Point, p2: &'a Point) -> &'a Point { if p1.0 < p2.0 { p1 } else { p2 } } fn main() { let p1: Point = Point(10, 10); let p3: &Point; { let p2: Point = Point(20, 20); p3 = left_most(&p1, &p2); } println!("left-most point: {:?}", p3); }
p3
์ ์๋ช ์ดp2
๋ณด๋ค ๊ธธ๊ธฐ ๋๋ฌธ์ ์ด ์์ ๋ ์ปดํ์ผ๋์ง ์์์ ํ์ธํ์๊ธฐ ๋ฐ๋๋๋ค. -
์์ ๊ณต๊ฐ์ ์ด๊ธฐํ ํ ํ ํจ์ ์๊ทธ๋์ฒ๋ฅผ
fn left_most<'a, 'b>(p1: &'a Point, p2: &'a Point) -> &'b Point
๋ก ๋ณ๊ฒฝํด ๋ด ๋๋ค. ์ด ๊ฒฝ์ฐ'a
์'b
์ฌ์ด์ ๊ด๊ณ๊ฐ ๋ถ๋ถ๋ช ํ๊ธฐ ๋๋ฌธ์ ์ปดํ์ผ ๋์ง ์์ต๋๋ค. -
์ด ์๋ฌ๋ฅผ ์ค๋ช ํ๋ ๋ ๋ค๋ฅธ ๋ฐฉ๋ฒ์ ๋ค์๊ณผ ๊ฐ์ต๋๋ค:
- ์ด ํจ์๋ ๋ ๊ฐ์ ๋น๋ ค์, ์๋ก์ด ์ฐธ์กฐ๋ฅผ ๋ฐํํฉ๋๋ค.
- ์ด ๋ฐํ๋ ์ฐธ์กฐ๋ ๋ ์ ๋ ฅ ์ค ํ๋๋ก ๋ถํฐ ์์ผ ํฉ๋๋ค. (์๋๋ฉด ์ ์ญ ๋ณ์๋ก ๋ถํฐ)
- ๋ ์ ๋ ฅ ์ค ์ด๋ค ๊ฒ์ผ๊น์? ์ปดํ์ผ๋ฌ๋ ์ด๋ฅผ ์์์ผ ํฉ๋๋ค. ๊ทธ๋์ผ๋ง ํจ์ ํธ์ถ๋ถ์์ ๋ดค์ ๋, ๋ฐํ๋ ์ฐธ์กฐ์ ์๋ช ์ด ์๋ ๊ฐ์ ์๋ช ๋ณด๋ค ๊ธธ์ง ์์์ ํ์ธํ ์ ์๊ธฐ ๋๋ฌธ์ ๋๋ค.
๊ตฌ์กฐ์ฒด์์์ ์๋ช
์ด๋ค ํ์ ์ด ๋น๋ ค์จ ๋ฐ์ดํฐ๋ฅผ ์ ์ฅํ๊ณ ์๋ค๋ฉด, ๋ฐ๋์ ์๋ช ์ ํ์ํด์ผ ํฉ๋๋ค:
#[derive(Debug)] struct Highlight<'doc>(&'doc str); fn erase(text: String) { println!("Bye {text}!"); } fn main() { let text = String::from("The quick brown fox jumps over the lazy dog."); let fox = Highlight(&text[4..19]); let dog = Highlight(&text[35..43]); // erase(text); println!("{fox:?}"); println!("{dog:?}"); }
- ์์ ์์ ์์
Highlight
์ ์ด๋ ธํ ์ด์ (<'doc>
)์ ์ ์ด๋Highlight
์ธ์คํด์ค๊ฐ ์ด์์๋ ๋์์๋ ๊ทธ ๋ด๋ถ์&str
๊ฐ ๊ฐ๋ฆฌํค๋ ๋ฐ์ดํฐ ์ญ์ ์ด์์์ด์ผ ํ๋ค๋ ๊ฒ์ ์๋ฏธํฉ๋๋ค. - ๋ง์ฝ
text
๊ฐfox
(ํน์dog
)์ ์๋ช ์ด ๋คํ๊ธฐ ์ ์erase
ํจ์ ํธ์ถ ๋ฑ์ผ๋ก ์ฌ๋ผ์ง๊ฒ ๋๋ค๋ฉด ๋น๋ฆผ ๊ฒ์ฌ๊ธฐ๊ฐ ์๋ฌ๋ฅผ ๋ฐ์ํฉ๋๋ค. - ๋น๋ฆฐ ๋ฐ์ดํฐ๋ฅผ ๊ฐ์ง๊ณ ์๋ ํ์ ์ ์ฌ์ฉ์๋ก ํ์ฌ๊ธ ์๋ณธ ๋ฐ์ดํฐ๋ฅผ ์ ์งํ๋๋ก ๊ฐ์ ํฉ๋๋ค. ์ด๋ฐ ํ์ ์ ๊ฒฝ๋ ๋ทฐ(lightweight view)๋ฅผ ๋ง๋๋๋ฐ ์ ์ฉํ์ง๋ง, ์ด ์ ์ฝ ์กฐ๊ฑด ๋๋ฌธ์ ์ด๋ฐ ํ์ ์ ์ฌ์ฉํ๋ ๊ฒ์ด ์ฝ์ง๋ง์ ์์ต๋๋ค.
- ๋ฐ๋ผ์, ๊ฐ๋ฅํ๋ค๋ฉด, ๊ตฌ์กฐ์ฒด๊ฐ ์์ ์ ๋ฐ์ดํฐ๋ฅผ ์ง์ ์์ ํ๋๋ก ํ๋ ๊ฒ์ด ์ข์ต๋๋ค.
- ํ ๊ตฌ์กฐ์ฒด์์ ์ฌ๋ฌ ์ฐธ์กฐ๊ฐ ์์ผ๋ฉด์, ์ด ์ฐธ์กฐ๋ค์ ์๋ช ์ด ์๋ก ๋ค๋ฅด๊ฒ ์ง์ ๋๋ ๊ฒฝ์ฐ๋ ์์ต๋๋ค. ์ด๋ ์ฐธ์กฐ์ ๊ทธ ๊ตฌ์กฐ์ฒด ๊ฐ์ ๊ด๊ณ ๋ฟ๋ง์ด ์๋๋ผ, ๊ทธ ์ฐธ์กฐ๋ค ์ฌ์ด์ ์๋ช ๊ด๊ณ๋ฅผ ์ค๋ช ํด์ผ ํ ๊ฒฝ์ฐ์ ํ์ํฉ๋๋ค. ๋งค์ฐ ๊ณ ๊ธ ๊ธฐ์ ์ ๋๋ค.
๊ตฌ์กฐ์ฒด
C/C++ ์ ๋ง์ฐฌ๊ฐ์ง๋ก ๋ฌ์คํธ๋ ์ปค์คํ ๊ตฌ์กฐ์ฒด๋ฅผ ์ง์ํฉ๋๋ค:
struct Person { name: String, age: u8, } fn main() { let mut peter = Person { name: String::from("Peter"), age: 27, }; println!("{} is {} years old", peter.name, peter.age); peter.age = 28; println!("{} is {} years old", peter.name, peter.age); let jackie = Person { name: String::from("Jackie"), ..peter }; println!("{} is {} years old", jackie.name, jackie.age); }
ํค ํฌ์ธํธ:
- ๊ตฌ์กฐ์ฒด๋ C/C++ ์ ์ ์ฌํฉ๋๋ค.
- C++ ์ ๊ฐ์ง๋ง C์๋ ๋ฌ๋ฆฌ ํ์ ์ ์ ์ํ๊ธฐ ์ํด โtypedefโ๊ฐ ํ์ํ์ง ์์ต๋๋ค.
- C++ ์ ๋ฌ๋ฆฌ ๊ตฌ์กฐ์ฒด ๊ฐ ์์์ ์์ต๋๋ค.
- ๋ฉ์๋๋
impl
๋ธ๋ก์ ์ ์ ํฉ๋๋ค. ๋ค์ ์ฌ๋ผ์ด๋์์ ํ์ธ ํ ์ ์์ต๋๋ค. - ์ฌ๋๋ค์๊ฒ ๋ค๋ฅธ ์ข
๋ฅ์ ๊ตฌ์กฐ์ฒด๊ฐ ์์์ ์๊ฒ ํ๊ธฐ์ ์ข์ ์๊ฐ์ผ ๊ฒ์
๋๋ค.
- 0 ํฌ๊ธฐ ๊ตฌ์กฐ์ฒด(์:
struct Foo;
)๋ ๋ฐ์ดํฐ๋ฅผ ๊ฐ์ง๊ณ ์์ง ์์ง๋ง ํน์ ํ์ ์ ํธ๋ ์์ ๊ตฌํํ ๋ ์ ์ฉํฉ๋๋ค. - ๋ค์ ์ฌ๋ผ์ด๋์์๋ ํ๋ ์ด๋ฆ์ด ๋ ์ค์ํ ๋ ์ฌ์ฉํ ์ ์๋ ํํ ๊ตฌ์กฐ์ฒด๋ฅผ ์๊ฐํฉ๋๋ค.
- 0 ํฌ๊ธฐ ๊ตฌ์กฐ์ฒด(์:
..peter
๋ฌธ๋ฒ์ ํ ๊ตฌ์กฐ์ฒด์์ ๋ค๋ฅธ ๊ตฌ์กฐ์ฒด๋ก ๋๋ถ๋ถ์ ๊ฐ์ ๋ณต์ฌํ๋ ค๊ณ ํ๋ ๊ฒฝ์ฐ์ ํ๋ํ๋ ํ์ดํํ๋ ์๊ณ ๋ฅผ ๋์ด์ค๋๋ค. ๋ฐ๋์ ๋งจ ๋ง์ง๋ง์ ์์ผ ํฉ๋๋ค.
ํํ
๊ฐ ํ๋ ์ด๋ฆ์ด ์ค์ํ์ง ์๋ค๋ฉด ํํ ๊ตฌ์กฐ์ฒด๋ฅผ ์ฌ์ฉํ ์ ์์ต๋๋ค:
struct Point(i32, i32); fn main() { let p = Point(17, 23); println!("({}, {})", p.0, p.1); }
ํํ ๊ตฌ์กฐ์ฒด๋ ์ข ์ข ๋จ์ผ ํ๋์ ๋ํผ(wrapper, ๋ฌ์คํธ์์ ๋ดํ์ (newtype)์ด๋ผ๊ณ ๋ถ๋ฆ)๋ก ์ฌ์ฉ๋ฉ๋๋ค:
struct PoundsOfForce(f64); struct Newtons(f64); fn compute_thruster_force() -> PoundsOfForce { todo!("Ask a rocket scientist at NASA") } fn set_thruster_force(force: Newtons) { // ... } fn main() { let force = compute_thruster_force(); set_thruster_force(force); }
- ๋ดํ์
์ ๊ธฐ๋ณธ ํ์
์ ๋ถ๊ฐ์ ์ธ ์๋ฏธ๋ฅผ ๋ํ๋ ์ข์ ๋ฐฉ๋ฒ์
๋๋ค. ์๋ฅผ ๋ค์ด:
- ์ซ์๊ฐ์ ๋จ์๋ฅผ ํ์ํ ์ ์์: ์์์
Newtons
์ด ๊ทธ ์์ ๋๋ค. - ๊ฐ์ด ์์ฑ๋ ๋ ์ด๋ฏธ ์ ํจ์ฑ ๊ฒ์ฌ๋ฅผ ํต๊ณผ ํ์ผ๋ฏ๋ก ์ถ๊ฐ์ ์ธ ๊ฒ์ฌ๊ฐ ํ์์์ต๋๋ค:
PhoneNumber(String)
๋๋OddNumber(u32)
.
- ์ซ์๊ฐ์ ๋จ์๋ฅผ ํ์ํ ์ ์์: ์์์
Newtons
ํ์ ์ ๊ฐ์f64
๊ฐ์ ๋ํ๋ ๋ฐฉ๋ฒ์ ๋ณด์ฌ์ฃผ์ธ์.- ๋ฌ์คํธ๋ ๋ถ๋ช ํ์ง ์์ ๊ฒ์ ์ซ์ดํฉ๋๋ค. ์๋ฅผ ๋ค๋ฉด ์๋์ผ๋ก unwrapํ๊ฑฐ๋ ๋ถ๋ฆฌ์ธ ๊ฐ์ ์ ์ ๊ฐ์ผ๋ก ์ฌ์ฉํ๋ ๊ฒ๋ค์ด ๊ทธ๋ ์ต๋๋ค.
- ์ฐ์ฐ์ ์ฌ์ ์๋ 3์ผ์ฐจ ์ ๋ค๋ฆญ ๋ถ๋ถ์์ ๋ค๋ฃน๋๋ค.
- ์ด๋ ํ์ฑ ๊ธฐํ ๊ถค๋์ (Mars Climate Orbiter)์ ์คํจ ์์ธ์ผ๋ก ์ง๋ชฉ๋ ๋๋ํ ์ ๋ ฅ ์ค๋ฅ๋ฅผ ๋ณด์ฌ์ค๋๋ค.
ํ๋ ํ ๋น ๋จ์ถ ๋ฌธ๋ฒ
๊ตฌ์กฐ์ฒด ํ๋์ ๋์ผํ ์ด๋ฆ์ ๋ณ์๊ฐ ์๋ค๋ฉด ์๋์ ๊ฐ์ด โ์งง์ ๋ฌธ๋ฒโ์ผ๋ก ๊ตฌ์กฐ์ฒด๋ฅผ ์์ฑํ ์ ์์ต๋๋ค:
#[derive(Debug)] struct Person { name: String, age: u8, } impl Person { fn new(name: String, age: u8) -> Person { Person { name, age } } } fn main() { let peter = Person::new(String::from("Peter"), 27); println!("{peter:?}"); }
-
new
ํจ์๋ฅผ ๋ค์์ฒ๋ผ ๊ตฌ์กฐ์ฒด ์ด๋ฆ ๋์Self
๋ฅผ ์ฌ์ฉํ์ฌ ์์ฑํด๋ ๋ฉ๋๋ค#[derive(Debug)] struct Person { name: String, age: u8, } impl Person { fn new(name: String, age: u8) -> Self { Self { name, age } } }
-
Default
ํธ๋ ์์ ๊ตฌํํด๋ณด์ธ์. ํ๋ ๋ช๊ฐ๋ ์ด๊ธฐํํ๊ณ ๋๋จธ์ง ํ๋๋ ๋ํดํธ ๊ฐ์ ์ฌ์ฉํ ์ ์์ต๋๋ค.#[derive(Debug)] struct Person { name: String, age: u8, } impl Default for Person { fn default() -> Person { Person { name: "Bot".to_string(), age: 0, } } } fn create_default() { let tmp = Person { ..Person::default() }; let tmp = Person { name: "Sam".to_string(), ..Person::default() }; }
-
๋ฉ์๋๋
impl
๋ธ๋ก์ ์ ์๋ฉ๋๋ค. -
peter
์ ๊ตฌ์กฐ์ฒด ์ ๋ฐ์ดํธ ๋ฌธ๋ฒ์ ์ฌ์ฉํ์ฌ ์๋ก์ด ๊ตฌ์กฐ์ฒด ์ธ์คํด์ค๋ฅผ ๋ง๋ค์ด๋ณด์ธ์. ์ด๋,peter
๋ ๋์ด์ ์ฌ์ฉํ ์ ์๊ฒ ๋ฉ๋๋ค. -
๊ตฌ์กฐ์ฒด๋ฅผ
Debug
ํํ๋ก ์ถ๋ ฅํ๋ ค๋ฉด{:#?}
๋ฅผ ์ฌ์ฉํ์ธ์.
๋ฉ์๋
๋ฌ์คํธ์์ ์ ์ธ๋ ํ์
์ ๋ํด impl
๋ธ๋ก์ ํจ์๋ฅผ ์ ์ธํ์ฌ ๋ฉ์๋๋ฅผ ์ฐ๊ฒฐ ํ ์ ์์ต๋๋ค:
#[derive(Debug)] struct Person { name: String, age: u8, } impl Person { fn say_hello(&self) { println!("Hello, my name is {}", self.name); } } fn main() { let peter = Person { name: String::from("Peter"), age: 27, }; peter.say_hello(); }
ํค ํฌ์ธํธ:
- ๋ฉ์๋๋ฅผ ํจ์์ ๋น๊ตํ์ฌ ์๊ฐํ๋ ๊ฒ๋ ๋์์ด ๋ ์ ์์ต๋๋ค.
- ๋ฉ์๋๋ ๊ตฌ์กฐ์ฒด๋ ์ด๊ฑฐํ๊ณผ ๊ฐ์ ํ์
์ ์ธ์คํด์ค์์ ํธ์ถ ๋๋ฉฐ, ์ฒซ๋ฒ์งธ ๋งค๊ฐ๋ณ์(ํ๋ผ๋ฉํฐ)๋ ์ธ์คํด์ค๋ฅผ
self
๋ก ํ๊ธฐํฉ๋๋ค. - ๋ฉ์๋๋ฅผ ์ด์ฉํ๋ฉด receiver ๋ฌธ๋ฒ์ ์ฌ์ฉํ ์ ์๊ณ ์ฝ๋๋ฅผ ์ข๋ ์ฒด๊ณ์ ์ผ๋ก ์ ๋ฆฌํ ์ ์์ต๋๋ค. ๋ฉ์๋๋ค์ด ์์ธก ๊ฐ๋ฅํ ์์น์ ๋ชจ์ฌ ์์ผ๋ ์ฐพ๊ธฐ ์ฝ์ต๋๋ค.
- ๋ฉ์๋๋ ๊ตฌ์กฐ์ฒด๋ ์ด๊ฑฐํ๊ณผ ๊ฐ์ ํ์
์ ์ธ์คํด์ค์์ ํธ์ถ ๋๋ฉฐ, ์ฒซ๋ฒ์งธ ๋งค๊ฐ๋ณ์(ํ๋ผ๋ฉํฐ)๋ ์ธ์คํด์ค๋ฅผ
- ๋ฉ์๋ receiver์ธ
self
ํค์๋ ์ฌ์ฉ์ ์ธ๊ธํด ์ฃผ์๊ธฐ ๋ฐ๋๋๋ค.- ์์ ์ ๊ฒฝ์ฐ
self: &Self
์ ์ค์ธ ๋ฒ์ ์์ ์๋ ค์ฃผ๊ณ , ๊ตฌ์กฐ์ฒด์ ์ด๋ฆ์ ์ง์ ์ฌ์ฉํ๋ฉด ์ด๋ป๊ฒ ๋๋์ง ๋ณด์ฌ์ฃผ๋ ๊ฒ๋ ์ข์ต๋๋ค. impl
๋ธ๋ก ๋ด๋ถ์์๋Self
๊ฐ ํด๋น ํ์ ์ ์ด๋ฆ ๋์ฉ์ผ๋ก ์ฌ์ฉ๋ ์ ์์์ ์๋ ค์ฃผ์ธ์.- ๊ตฌ์กฐ์ฒด์ ํ๋๋ฅผ ์ ๊ทผํ ๋ ์ ํ๊ธฐ๋ฅผ ์ฌ์ฉํ๋ฏ์ด
self
์ ์ ํ๊ธฐ๋ฅผ ์ฌ์ฉํ์ฌ ๊ฐ๋ณ ํ๋๋ค์ ์ ๊ทผํ ์ ์์ต๋๋ค. say_hello
ํจ์๊ฐ ๋ ๋ฒ ํธ์ถ๋๋๋ก ์ฝ๋๋ฅผ ์์ ํ์ฌ&self
์self
๊ฐ ์ด๋ป๊ฒ ๋ค๋ฅธ์ง ๋ณด์ฌ์ฃผ๋ ๊ฒ๋ ์ข์ต๋๋ค.
- ์์ ์ ๊ฒฝ์ฐ
- ๋ค์ ์ฌ๋ผ์ด๋์์ receiver์ ๊ตฌ๋ถ์ ์ค๋ช ํฉ๋๋ค.
๋ฉ์๋ ๋ฆฌ์๋ฒ(Receiver)
&self
๋ ๋ฉ์๋๊ฐ ๊ฐ์ฒด๋ฅผ ๋ถ๋ณํ๊ฒ ๋น๋ ค์ด์ ๋ํ๋
๋๋ค. ๋ฉ์๋์ ๋ฆฌ์๋ฒ๋ ๋ค์์ ํํ๋ค์ด ๊ฐ๋ฅํฉ๋๋ค:
&self
: ํธ์ถ์๋ก๋ถํฐ ๊ณต์ ๊ฐ๋ฅํ ๋ถ๋ณ ์ฐธ์กฐ ๋ฐฉ์์ผ๋ก ๊ฐ์ฒด๋ฅผ ๋น๋ ค์ด์ ๋ํ๋ ๋๋ค. ๊ฐ์ฒด๋ ๋ฉ์๋ ํธ์ถ ๋ค์๋ ์ฌ์ฉ๋ ์ ์์ต๋๋ค.&mut self
: ํธ์ถ์๋ก๋ถํฐ ์ ์ผํ ๊ฐ๋ณ ์ฐธ์กฐ ๋ฐฉ์์ผ๋ก ๊ฐ์ฒด๋ฅผ ๋น๋ ค์ด์ ๋ํ๋ ๋๋ค. ๊ฐ์ฒด๋ ๋ฉ์๋ ํธ์ถ ๋ค์๋ ์ฌ์ฉ๋ ์ ์์ต๋๋ค.self
: ํธ์ถ์๋ก๋ถํฐ ๊ฐ์ฒด์ ์์ ๊ถ์ ๊ฐ์ ธ์ค๊ณ ๊ฐ์ฒด๋ ํธ์ถ์๋ก๋ถํฐ ๋ฉ์๋๋ก ์ด๋๋ฉ๋๋ค. ๋ฉ์๋๊ฐ ๊ฐ์ฒด๋ฅผ ์์ ํ๊ฒ ๋๋ฉฐ ๋ฐ๋ผ์ ๋ช ์์ ์ผ๋ก ์์ ๊ถ์ ๋ค๋ฅธ ๊ณณ์ผ๋ก ์ ๋ฌํ์ง ์๋๋ค๋ฉด ๋ฉ์๋ ์ข ๋ฃ์ ํจ๊ป ๊ฐ์ฒด๋ drop(ํด์ )๋ฉ๋๋ค.mut self
: ์์ ๋์ผํ์ง๋ง ๋ฉ์๋๊ฐ ๊ฐ์ฒด์ ์์ ๊ถ์ ๊ฐ์ง๋ฉด์ ๋์์ ๊ฐ์ฒด๋ฅผ ์์ ํ ์๋ ์์ต๋๋ค. ์์ ๊ถ์ ๊ฐ์ง๋ ๊ฒ์ด ์์ ํ ์ ์์์ ์๋ฏธํ๋ ๊ฒ์ ์๋๋๋ค.- ๋ฆฌ์๋ฒ ์์: ๊ตฌ์กฐ์ฒด์ ์ ์ ๋ฉ์๋๊ฐ ๋ฉ๋๋ค. ์ฃผ๋ก ์์ฑ์๋ฅผ ๋ง๋ค๋ ์ฌ์ฉํ๊ฒ ๋๋ฉฐ, ์์ฑ์๋ ํํ
new
๋ผ๊ณ ์ด๋ฆ๋ถ์ ๋๋ค.
self
๋ฅผ ์ฌ์ฉํ๋ ์ด๊ฐ์ ๋ณํ๋ค ์ธ์๋ Box<Self>
์ ๊ฐ์ด ๋ฆฌ์๋ฒ ํ์
์ผ๋ก ํ์ฉ๋๋ ํน๋ณํ ๋ํผ ํ์
์ด ์์ต๋๋ค.
โ๊ณต์ ๊ฐ๋ฅํ ๋ถ๋ณโ๊ณผ โ์ ์ผํ ๊ฐ๋ณโ ๋ถ๋ถ์ ๊ฐ์กฐํ ๋งํฉ๋๋ค. ์ด๋ฌํ ์ ์ฝ์ ๋ฌ์คํธ์ ๋น๋ฆผ ๊ฒ์ฌ๊ธฐ(borrow checker) ๊ท์น์ผ๋ก ๋ ๋ถ์ด๋ค๋๋๋ค. self
๋ ์์ธ๋ ์๋๋๋ค. ์ฌ๋ฌ ์์น์์ ๊ตฌ์กฐ์ฒด๋ฅผ ์ฐธ์กฐํ๋ฉด์ ๊ฐ์ฒด๋ฅผ ์์ ํ๋(&mut self
๋ฅผ ๋ฆฌ์๋ฒ๋ก ํ๋) ๋ฉ์๋๋ฅผ ํธ์ถํ๋ ๊ฒ์ ๋ถ๊ฐ๋ฅํฉ๋๋ค.
์์
#[derive(Debug)] struct Race { name: String, laps: Vec<i32>, } impl Race { fn new(name: &str) -> Race { // No receiver, a static method Race { name: String::from(name), laps: Vec::new() } } fn add_lap(&mut self, lap: i32) { // Exclusive borrowed read-write access to self self.laps.push(lap); } fn print_laps(&self) { // Shared and read-only borrowed access to self println!("Recorded {} laps for {}:", self.laps.len(), self.name); for (idx, lap) in self.laps.iter().enumerate() { println!("Lap {idx}: {lap} sec"); } } fn finish(self) { // Exclusive ownership of self let total = self.laps.iter().sum::<i32>(); println!("Race {} is finished, total lap time: {}", self.name, total); } } fn main() { let mut race = Race::new("Monaco Grand Prix"); race.add_lap(70); race.add_lap(68); race.print_laps(); race.add_lap(71); race.print_laps(); race.finish(); // race.add_lap(42); }
ํค ํฌ์ธํธ:
- ์ด ๋ค ๊ฐ์ ๋ฉ์๋๋ ์๋ก ๋ค๋ฅธ ์ ํ์ ๋ฆฌ์๋ฒ๋ฅผ ์ฌ์ฉํฉ๋๋ค.
- ๋ฆฌ์๋ฒ์ ์ ํ์ ๋ฐ๋ผ ํจ์๊ฐ ํ ์ ์๋ ์ผ์ด ๋ฌ๋ผ์ง๊ณ , ๋ ๋ฉ์๋๋ฅผ ํธ์ถํ ๋ค
main
์์ ํด๋น ๊ฐ์ฒด๋ฅผ ์ฌ์ฉํ ์ ์๋์ง ์ฌ๋ถ๋ ๋ฌ๋ผ์ง๋ค๋ ์ ์ ๊ฐ์กฐํ์ธ์. finish
๋ฅผ ๋๋ฒ ํธ์ถํ์ฌ ์ค๋ฅ๊ฐ ๋ฐ์ํ๋ ๊ฒ์ ๋ณด์ผ ์ ์์ต๋๋ค.
- ๋ฆฌ์๋ฒ์ ์ ํ์ ๋ฐ๋ผ ํจ์๊ฐ ํ ์ ์๋ ์ผ์ด ๋ฌ๋ผ์ง๊ณ , ๋ ๋ฉ์๋๋ฅผ ํธ์ถํ ๋ค
- ๋น๋ก ๋ฉ์๋ receiver๋ ๋ค๋ฅด์ง๋ง main ํจ์์์ ๋น ์ ์ ํจ์๋ฅผ ๋ถ๋ฅด๋ ๋ฐฉ๋ฒ์ ๊ฐ์ต๋๋ค. ๋ฌ์คํธ๋ ๋ฉ์๋๋ฅผ ํธ์ถํ ๋ ์๋์ผ๋ก ์ฐธ์กฐ/์ญ์ฐธ์กฐ(๋ฐ๋ผ๊ฐ๊ธฐ)๋ฅผ ์ํํฉ๋๋ค. ๋ฌ์คํธ๋ ๊ฐ์ฒด์ ๋งค์๋ ์๊ทธ๋์ฒ๊ฐ ์๋ก ๋งค์น๋๋๋ก ๊ฐ์ฒด์
&
,*
,muts
๋ฅผ ์๋์ผ๋ก ๋ถ์ฌ์ค๋๋ค. print_laps
ํจ์์์ ๋ฒกํฐ๋ฅผ ์ด๋ค ์์ผ๋ก ์ฌ์ฉํ๊ณ ์๋์ง ์ธ๊ธํ๋ ๊ฒ๋ ์ข์ต๋๋ค. ๋ฒกํฐ๋ ์คํ ๊ฐ์์์ ๋ ์์ธํ ์ค๋ช ํ ๊ฒ์ ๋๋ค.
2์ผ์ฐจ ์ค์ ์ฐ์ต๋ฌธ์
์ด๋ฒ ์ฐ์ต๋ฌธ์ ๋ค์ ๋๊ฐ์ง ๋งฅ๋ฝ์์ ๋ฉ์๋ ๊ตฌํ๋ฐฉ๋ฒ์ ๋ค๋ฃน๋๋ค:
-
Storing books and querying the collection
-
Keeping track of health statistics for patients
After looking at the exercises, you can look at the solutions provided.
์ฑ ์ ์ฅํ๊ธฐ
์ฐ๋ฆฌ๋ ๋ด์ผ ๊ตฌ์กฐ์ฒด์ Vec<T>
์ ๋ํด ๋ ๋ง์ ๊ฒ์ ๋ฐฐ์ธ ๊ฒ์
๋๋ค. ์ผ๋จ ์ค๋์ API์ ์ผ๋ถ๋ง ์๋ฉด ๋ฉ๋๋ค:
fn main() { let mut vec = vec![10, 20]; vec.push(30); let midpoint = vec.len() / 2; println!("middle value: {}", vec[midpoint]); for item in &vec { println!("item: {item}"); } }
์๋ ์ฝ๋๋ ๋์๊ด์ ์๋ ๋์ ์ปฌ๋์ ์ ๋ชจ๋ธ๋ง ํฉ๋๋ค. ์๋ ์ฝ๋๋ฅผ https://play.rust-lang.org/์ ๋ณต์ฌํ ํ, ์ปดํ์ผ ๋๋๋ก ์์ ํด ๋ด ์๋ค:
struct Library { books: Vec<Book>, } struct Book { title: String, year: u16, } impl Book { // This is a constructor, used below. fn new(title: &str, year: u16) -> Book { Book { title: String::from(title), year, } } } // Implement the methods below. Notice how the `self` parameter // changes type to indicate the method's required level of ownership // over the object: // // - `&self` for shared read-only access, // - `&mut self` for unique and mutable access, // - `self` for unique access by value. impl Library { fn new() -> Library { todo!("Initialize and return a `Library` value") } fn len(&self) -> usize { todo!("Return the length of `self.books`") } fn is_empty(&self) -> bool { todo!("Return `true` if `self.books` is empty") } fn add_book(&mut self, book: Book) { todo!("Add a new book to `self.books`") } fn print_books(&self) { todo!("Iterate over `self.books` and print each book's title and year") } fn oldest_book(&self) -> Option<&Book> { todo!("Return a reference to the oldest book (if any)") } } fn main() { let mut library = Library::new(); println!( "The library is empty: library.is_empty() -> {}", library.is_empty() ); library.add_book(Book::new("Lord of the Rings", 1954)); library.add_book(Book::new("Alice's Adventures in Wonderland", 1865)); println!( "The library is no longer empty: library.is_empty() -> {}", library.is_empty() ); library.print_books(); match library.oldest_book() { Some(book) => println!("The oldest book is {}", book.title), None => println!("The library is empty!"), } println!("The library has {} books", library.len()); library.print_books(); }
๊ฑด๊ฐ์ํ ๋ชจ๋ํฐ๋ง ์์คํ
๋น์ ์ ๊ฑด๊ฐ ์ํ๋ฅผ ๋ชจ๋ํฐ๋งํ๋ ์์คํ ์ ๊ตฌํํ๋ ์ผ์ ํ๊ณ ์์ต๋๋ค. ๊ทธ ์ผํ์ผ๋ก ๋น์ ์ ์ฌ์ฉ์์ ๊ฑด๊ฐ ์ํ ํต๊ณ๋ฅผ ์ถ์ ํด์ผํฉ๋๋ค.
๋น์ ์ ๋ชฉํ๋ User
๊ตฌ์กฐ์ฒด์ impl
๋ธ๋ก์ ๋น ํจ์๋ฅผ ๊ตฌํํ๋ ๊ฒ์
๋๋ค.
์๋ ์ฝ๋๋ฅผ https://play.rust-lang.org/์ ๋ณต์ฌํด์ ๋น ์ง ๋ฉ์๋๋ฅผ ๊ตฌํํ๋ฉด ๋ฉ๋๋ค:
// TODO: remove this when you're done with your implementation. #![allow(unused_variables, dead_code)] pub struct User { name: String, age: u32, height: f32, visit_count: usize, last_blood_pressure: Option<(u32, u32)>, } pub struct Measurements { height: f32, blood_pressure: (u32, u32), } pub struct HealthReport<'a> { patient_name: &'a str, visit_count: u32, height_change: f32, blood_pressure_change: Option<(i32, i32)>, } impl User { pub fn new(name: String, age: u32, height: f32) -> Self { unimplemented!() } pub fn name(&self) -> &str { unimplemented!() } pub fn age(&self) -> u32 { unimplemented!() } pub fn height(&self) -> f32 { unimplemented!() } pub fn doctor_visits(&self) -> u32 { unimplemented!() } pub fn set_age(&mut self, new_age: u32) { unimplemented!() } pub fn set_height(&mut self, new_height: f32) { unimplemented!() } pub fn visit_doctor(&mut self, measurements: Measurements) -> HealthReport { unimplemented!() } } fn main() { let bob = User::new(String::from("Bob"), 32, 155.2); println!("I'm {} and my age is {}", bob.name(), bob.age()); } #[test] fn test_height() { let bob = User::new(String::from("Bob"), 32, 155.2); assert_eq!(bob.height(), 155.2); } #[test] fn test_set_age() { let mut bob = User::new(String::from("Bob"), 32, 155.2); assert_eq!(bob.age(), 32); bob.set_age(33); assert_eq!(bob.age(), 33); } #[test] fn test_visit() { let mut bob = User::new(String::from("Bob"), 32, 155.2); assert_eq!(bob.doctor_visits(), 0); let report = bob.visit_doctor(Measurements { height: 156.1, blood_pressure: (120, 80), }); assert_eq!(report.patient_name, "Bob"); assert_eq!(report.visit_count, 1); assert_eq!(report.blood_pressure_change, None); let report = bob.visit_doctor(Measurements { height: 156.1, blood_pressure: (115, 76), }); assert_eq!(report.visit_count, 2); assert_eq!(report.blood_pressure_change, Some((-5, -4))); }
ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ
๋ฌ์คํธ์์ ์ ๊ณตํ๋ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ ๋ฌ์คํธ๋ก ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ ํ๋ก๊ทธ๋จ์ ์์ฑํ ๋ ๊ณตํต์ ์ผ๋ก ์ฌ์ฉํ ์ ์๋ ์ฌ๋ฌ ํ์
๋ค์ ํฌํจํ๊ณ ์์ต๋๋ค. ์ด๋ฅผ ํตํด ์๋ก ๋ค๋ฅธ ๋ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ผ ํ๋๋ผ๋ ํจ๊ป ์ฌ์ฉํ๋๋ฐ ํฐ ์ด๋ ค์์ด ์๊ฒ ๋ฉ๋๋ค. ์๋ฅผ ๋ค๋ฉด ๋ ๋ผ์ด๋ธ๋ฌ๋ฆฌ ๋ชจ๋ ๊ฐ์ String
ํ์
์ ์ฌ์ฉํ๊ธฐ ๋๋ฌธ์
๋๋ค.
์ผ๋ฐ์ ์ธ ํ์ ์ ์๋์ ๊ฐ์ต๋๋ค:
-
Option
๊ณผResult
: ์ด๋ค ๊ฐ์ด ์๊ฑฐ๋ ์๊ฑฐ๋ ํ๋ ๊ฒฝ์ฐ, ๊ทธ๋ฆฌ๊ณ ์ค๋ฅ ์ฒ๋ฆฌ์ ์ฌ์ฉํฉ๋๋ค. -
String
: ๊ธฐ๋ณธ์ ์ธ ๋ฌธ์์ด ํ์ ์ผ๋ก, ๋ฌธ์์ด ๋ฐ์ดํฐ๋ฅผ ์์ ํ๋ ๊ฒฝ์ฐ์ ์ฌ์ฉํฉ๋๋ค. -
Vec
: ๊ฐ๋ณ ํฌ๊ธฐ์ ํ์ค ๋ฒกํฐ ํ์ ์ ๋๋ค. -
HashMap
: ํด์ ์๊ณ ๋ฆฌ์ฆ์ ๋ฐ๋ก ์ง์ ํ ์๋ ์๋ ํด์๋งต ํ์ ์ ๋๋ค. -
Box
: ํ ๋ฐ์ดํฐ์ ๋ํ ์์ ํฌ์ธํฐ์ ๋๋ค. -
Rc
: ํ์ ํ ๋น๋ ๋ฐ์ดํฐ์ ๋ํ ์ฐธ์กฐ ์นด์ดํ ๊ณต์ ํฌ์ธํฐ์ ๋๋ค.
- ์ฌ์ค, ๋ฌ์คํธ์ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋
core
,alloc
,std
์ ๊ฐ์ด ๊ณ์ธต(layer)์ผ๋ก ๋๋ ์ง๋๋ค. core
๋libc
๋ ํ ๋น์(allocator), ์ฌ์ง์ด OS์๋ ์์กดํ์ง ์๋ ๊ฐ์ฅ ๊ธฐ๋ณธ์ ์ธ ํจ์์ ํ์ ์ ํฌํจํฉ๋๋ค.alloc
์Vec
,Box
,Arc
์ ๊ฐ์ด ์ ์ญ ํ ํ ๋น์ด ํ์ํ ํ์ ์ ํฌํจํฉ๋๋ค.- ์๋ฒ ๋๋ ๋ฌ์คํธ ์์ฉํ๋ก๊ทธ๋จ์ ์ฃผ๋ก
core
๋ง ์ฌ์ฉํ๊ฑฐ๋ ๊ฐ๋alloc
์ ํจ๊ป ์ฌ์ฉํฉ๋๋ค.
Option
๊ณผ Result
์ด ํ์ ์ ์ ํ์ ๋ฐ์ดํฐ๋ฅผ ๋ํ๋ ๋๋ค:
fn main() { let numbers = vec![10, 20, 30]; let first: Option<&i8> = numbers.first(); println!("first: {first:?}"); let arr: Result<[i8; 3], Vec<i8>> = numbers.try_into(); println!("arr: {arr:?}"); }
Option
๊ณผResult
๋ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฟ๋ง์๋๋ผ ๋งค์ฐ ๊ด๋ฒ์ํ๊ฒ ์ฌ์ฉ๋๋ ํ์ ์ ๋๋ค.Option<&T>
๋&T
์ ๋นํด ๊ณต๊ฐ ์ค๋ฒํค๋๊ฐ ์์ต๋๋ค.Result
๋ ์ค๋ฅ ์ฒ๋ฆฌ๋ฅผ ์ํ ํ์ค ํ์ ์ ๋๋ค. 3์ผ์ฐจ ๊ณผ์ ์์ ์ดํด๋ด ๋๋ค.try_into
attempts to convert the vector into a fixed-sized array. This can fail:- If the vector has the right size,
Result::Ok
is returned with the array. - Otherwise,
Result::Err
is returned with the original vector.
- If the vector has the right size,
String
String
์ ํ์ ํ ๋น๋๊ณ ๊ฐ๋ณ ๊ธธ์ด์ ํ์ค UTF-8 ๋ฌธ์์ด ๋ฒํผ์
๋๋ค:
fn main() { let mut s1 = String::new(); s1.push_str("Hello"); println!("s1: len = {}, capacity = {}", s1.len(), s1.capacity()); let mut s2 = String::with_capacity(s1.len() + 1); s2.push_str(&s1); s2.push('!'); println!("s2: len = {}, capacity = {}", s2.len(), s2.capacity()); let s3 = String::from("๐จ๐ญ"); println!("s3: len = {}, number of chars = {}", s3.len(), s3.chars().count()); }
String
์ Deref<Target = str>
์ ๊ตฌํํฉ๋๋ค. ์ด๋ , String
๊ฐ์ ๋ํด์๋ str
์ ๋ชจ๋ ๋ฉ์๋๋ฅผ ํธ์ถ ํ ์ ์๋ค๋ ์๋ฏธ ์
๋๋ค.
String::new
๋ ์๋ก์ด ๋น ๋ฌธ์์ด์ ๋ฐํํฉ๋๋ค.String::with_capacity
๋ ์๋ก ๋ง๋ค ๋ฌธ์์ด ๋ฒํผ์ ๋ฃ์ ๋ฐ์ดํฐ ํฌ๊ธฐ๋ฅผ ์๊ณ ์๋ ๊ฒฝ์ฐ์ ์ฌ์ฉํ ์ ์์ต๋๋ค.String::len
์String
์ ๋ฐ์ดํธ ํฌ๊ธฐ๋ฅผ ๋ฐํํฉ๋๋ค. (์ค์ ๋ฌธ์ ๊ฐ์์๋ ๋ค๋ฅผ ์ ์์ต๋๋ค.)String::chars
๋ ์ค์ ๋ฌธ์(character)๋ค์ ๋ํ ์ดํฐ๋ ์ดํฐ๋ฅผ ๋ฐํํฉ๋๋ค.char
๋ก ํํ๋๋ ๋ฌธ์๋ ์ฐ๋ฆฌ๊ฐ ์ค์ ๋ก ์ธ์ํ๊ณ ์ฌ์ฉํ๋ ๋ฌธ์์๋ ๋ค๋ฅผ ์ ์์ต๋๋ค. ์์ ๊ฒฐํฉ์ผ๋ก ๋ฌธ์๋ฅผ ํํํ๋ ๊ฒฝ์ฐ๊ฐ ์๊ธฐ ๋๋ฌธ์ ๋๋ค. ์ด์ ๋ํด์๋ Grapheme Cluster๋ฅผ ์ฐธ๊ณ ํ์ธ์.- ์ฌ๋๋ค์ด ๋ฌธ์์ด์ด๋ผ๊ณ ๋งํ ๋์๋
&str
์ด๊ฑฐ๋String
์ผ ์ ์์ต๋๋ค. - ์ด๋ค ํ์
์ด
Deref<Target = T>
๋ฅผ ๊ตฌํํ๊ณ ์์ผ๋ฉด, ์ปดํ์ผ๋ฌ๋ ์ฌ๋ฌ๋ถ์ดT
์ ๋ฉ์๋๋ค์ ํธ์ถํ ์ ์๊ฒ ๋์์ค๋๋ค.String
์Deref<Target = str>
์ ๊ตฌํํ๊ณ ์๊ธฐ ๋๋ฌธ์String
์ ๋ํด์๋str
๋ฉ์๋๋ค์ ํธ์ถํ ์ ์์ต๋๋ค.let s3 = s1.deref();
์let s3 = &*s1;
์ ๋น๊ตํด๋ณด์ธ์.
String
์ ๋ฐ์ดํธ ๋ฒกํฐ์ ๋ํผ๋ก ๊ตฌํ๋์ด ์์ต๋๋ค. ๋ฒกํฐ๊ฐ ์ง์ํ๋ ์ฌ๋ฌ๊ฐ์ง ์ฐ์ฐ๋ค์String
๋ ์ง์ํฉ๋๋ค. ๋ค๋งString
์ ๋ช๊ฐ์ง ๋ณด์ฅ ๋ด์ฉ์ด ๋ ์์ต๋๋ค.String
์ ์ธ๋ฑ์ค๋ก ์ ๊ทผํ๋ ๋ฐฉ๋ฒ๋ค์ ๋น๊ตํด๋ณด์ธ์:s3.chars().nth(i).unwrap()
๋ฅผ ์ด์ฉํ์ฌ ํ ๋ฌธ์๋ฅผ ์ ํํ๋ ๊ฒฝ์ฐ,i
๊ฐ์ด ๋ฒ์๋ฅผ ๋ฒ์ด๋ ๋, ๋ฒ์ด๋์ง ์์ ๋ ๋์์ ์ค๋ช ํ์ธ์.s3[0..4]
๋ฅผ ์ด์ฉํด์ ๋ฌธ์์ด์ ์ผ๋ถ๋ฅผ ์ ํํ๋๋ฐ, ์ฌ๋ผ์ด์ค๊ฐ ์ ๋์ฝ๋ ๋ฌธ์์ด ๊ฒฝ๊ณ์ ๋ฑ ๋ง์ง ์์ ๊ฒฝ์ฐ ์ด๋ป๊ฒ ๋๋์ง ์ค๋ช ํ์ธ์.
Vec
Vec
๋ ํ์ ํ ๋น๋ ํ์ค ๊ฐ๋ณ ํฌ๊ธฐ ๋ฒํผ์
๋๋ค:
fn main() { let mut v1 = Vec::new(); v1.push(42); println!("v1: len = {}, capacity = {}", v1.len(), v1.capacity()); let mut v2 = Vec::with_capacity(v1.len() + 1); v2.extend(v1.iter()); v2.push(9999); println!("v2: len = {}, capacity = {}", v2.len(), v2.capacity()); // Canonical macro to initialize a vector with elements. let mut v3 = vec![0, 0, 1, 2, 3, 4]; // Retain only the even elements. v3.retain(|x| x % 2 == 0); println!("{v3:?}"); // Remove consecutive duplicates. v3.dedup(); println!("{v3:?}"); }
Vec
์ Deref<Target = [T]>
๋ฅผ ๊ตฌํํฉ๋๋ค. ์ด๋ Vec
์์ ์ฌ๋ผ์ด์ค ๋ฉ์๋๋ฅผ ํธ์ถ ํ ์ ์๋ค๋ ์๋ฏธ์
๋๋ค.
Vec
์String
์ด๋HashMap
๊ณผ ๊ฐ์ ์ปฌ๋ ์ ํ์ ์ ๋๋ค. ๋ฒกํฐ๋ ๋ฐ์ดํฐ๋ฅผ ํ์ ์ ์ฅํฉ๋๋ค. ์ด๋ ์ปดํ์ผ ์์ ์ ๋ฐ์ดํฐ ํฌ๊ธฐ๋ฅผ ์ ํ์๊ฐ ์๋ค๋ ์๋ฏธ๊ณ , ๋ฐํ์์ ๋ ์ปค์ง ์๋ ์์์ง ์๋ ์์ต๋๋ค.Vec<T>
๋ ์ ๋ค๋ฆญ ํ์ ์ด๊ธฐ๋ ํฉ๋๋ค. ํ์ง๋งT
๋ฅผ ๊ผญ ์ง์ ํด์ค ํ์๋ ์์ต๋๋ค. ์ด ๊ฒฝ์ฐ, ๋ฌ์คํธ ํ์ ์ถ๋ก ์ด ๋ฒกํฐ์ ์ฒ์push
ํ๋ ๋ฐ์ดํฐ๋กT
๋ฅผ ์ ์ ์์์ต๋๋ค.vec![...]
๋Vec::new()
๋์ ์ธ ์ ์๋ ํ์ค ๋งคํฌ๋ก๋ก์, ์ด๊ธฐ ๋ฐ์ดํฐ๋ฅผ ์ถ๊ฐํ ๋ฒกํฐ๋ฅผ ์์ฑํ ์ ์์ต๋๋ค.- ๋ฒกํฐ๋
[
]
๋ฅผ ์ฌ์ฉํ์ฌ ์ธ๋ฑ์ค๋ก ์ ๊ทผํ ์ ์์ต๋๋ค. ํ์ง๋ง ๋ฒ์๋ฅผ ๋ฒ์ด๋๋ฉด ํจ๋์ด ๋ฐ์ํฉ๋๋ค. ๋์get
์ ์ฌ์ฉํ๋ฉดOption
์ ๋ฐํํฉ๋๋ค.pop
ํจ์๋ ๋ง์ง๋ง ์์๋ฅผ ์ ๊ฑฐํฉ๋๋ค. - ๋ฒกํฐ๋ฅผ ์ํํ๋ฉด์ ๊ฐ์ ๋ณ๊ฒฝํ ์๋ ์์์ ๋ณด์ฌ์ฃผ์ธ์:
for e in &mut v { *e += 50; }
HashMap
HashDoS ๊ณต๊ฒฉ์ผ๋ก๋ถํฐ ๋ณดํธ๋๋ ํ์ค ํด์ ๋งต์ ๋๋ค:
use std::collections::HashMap; fn main() { let mut page_counts = HashMap::new(); page_counts.insert("Adventures of Huckleberry Finn".to_string(), 207); page_counts.insert("Grimms' Fairy Tales".to_string(), 751); page_counts.insert("Pride and Prejudice".to_string(), 303); if !page_counts.contains_key("Les Misรฉrables") { println!("We know about {} books, but not Les Misรฉrables.", page_counts.len()); } for book in ["Pride and Prejudice", "Alice's Adventure in Wonderland"] { match page_counts.get(book) { Some(count) => println!("{book}: {count} pages"), None => println!("{book} is unknown.") } } // Use the .entry() method to insert a value if nothing is found. for book in ["Pride and Prejudice", "Alice's Adventure in Wonderland"] { let page_count: &mut i32 = page_counts.entry(book.to_string()).or_insert(0); *page_count += 1; } println!("{page_counts:#?}"); }
-
HashMap
์ prelude์ ์ ์๋์ด ์์ง ์๊ธฐ ๋๋ฌธ์ ๋ช ์์ ์ผ๋ก ์ถ๊ฐํด์ค์ผ ํฉ๋๋ค. -
์๋ ์ฝ๋๋ฅผ ํ ์คํธํด๋ณด์ธ์. ์ฒซ ๋ฌธ์ฅ์์๋ ํด์๋งต์ ์ฑ ์ด ์๋์ง ๊ฒ์ฌํ์ฌ, ์์ผ๋ฉด ๋ํดํธ ๊ฐ์ ๋ฐํํฉ๋๋ค. ๋๋ฒ ์งธ ๋ฌธ์ฅ์์๋ ํด์๋งต์ ํด๋น ์ฑ ์ด ์๋ ๊ฒฝ์ฐ, ์ง์ ํ ๊ฐ์ ํด์๋งต์ ์ถ๊ฐํ ๋ค ๊ทธ ๊ฐ์ ๋ฐํํฉ๋๋ค.
let pc1 = page_counts .get("Harry Potter and the Sorcerer's Stone ") .unwrap_or(&336); let pc2 = page_counts .entry("The Hunger Games".to_string()) .or_insert(374);
-
์ํ๊น์ง๋ง
hashmap!
๊ฐ์ ๋งคํฌ๋ก๊ฐ ์์ต๋๋ค.-
๋ฌ์คํธ 1.56๋ถํฐ๋
HashMap
์ดFrom<[(K, V); N]>
์ ๊ตฌํํ๊ธฐ ๋๋ฌธ์ ๋ฐฐ์ด ๋ฆฌํฐ๋ด์ ์ด์ฉํ์ฌ ์ฝ๊ฒ ํด์๋งต์ ์ด๊ธฐํํ ์ ์์ต๋๋ค:let page_counts = HashMap::from([ ("Harry Potter and the Sorcerer's Stone".to_string(), 336), ("The Hunger Games".to_string(), 374), ]);
-
-
ํค-๊ฐ ์์ ๋ํ
Iterator
๋ก ํด์๋งต์ ๋ง๋ค ์๋ ์์ต๋๋ค. -
์์ ์ฝ๋์์๋ ํธ์์ ํด์๋งต์ ํค๋ก
&str
๋ฅผ ์ฌ์ฉํ์ง ์์์ต๋๋ค. ๋ฌผ๋ก ์ปฌ๋ ์ ์ ์ฐธ์กฐ๋ฅผ ์ฌ์ฉํ ์๋ ์์ต๋๋ค. ๋ค๋ง ์ฐธ์กฐ๋ฅผ ์ฌ์ฉํ๊ฒ ๋๋ฉด ๋น๋ฆผ ๊ฒ์ฌ๊ธฐ ๋๋ฌธ์ ๋ณต์กํด ์ง ์ ์์ต๋๋ค.- ์์ ์ฝ๋์์
to_string()
์ ์์ ๋ ์ปดํ์ผ์ ๋ฌธ์ ๊ฐ ์๋์ง ํ์ธํด๋ณด์ธ์. ์ด๋ค ๋ฌธ์ ์ ๋ถ๋ชํ๊น์?
- ์์ ์ฝ๋์์
-
ํด์๋งต์ ๋ช ๋ช ๋ฉ์๋๋ ํด์๋งต ๋ด๋ถ์ ํน๋ณํ ํ์ (์๋ฅผ ๋ค์ด
std::collections::hash_map::Keys
)๋ค์ ๋ฆฌํดํฉ๋๋ค. ์ด๋ฌํ ํ์ ๋ค์ Rust ๋ฌธ์์์๋ ๊ฒ์ํ ์ ์์ต๋๋ค. ์๊ฐ์๋ค์๊ฒ ์ด ํ์ ๋ค์ ๋ํ ๋ฌธ์๋ฅผ ๋ณด์ฌ์ฃผ๊ณ , ์ด ๋ฌธ์์keys
๋ฉ์๋๋ก์ ์ญ ๋งํฌ๊ฐ ์์์ ์๋ ค์ฃผ์ธ์.
Box
Box
๋ ํ ๋ฐ์ดํฐ์ ๋ํ ์์ ํฌ์ธํฐ์
๋๋ค:
fn main() { let five = Box::new(5); println!("five: {}", *five); }
Box<T>
์ Deref<Target = T>
๋ฅผ ๊ตฌํํฉ๋๋ค. ์ด๋ Box<T>
์์ T
๋ฉ์๋๋ฅผ ์ง์ ํธ์ถ ํ ์ ์๋ค๋ ์๋ฏธ์
๋๋ค.
Box
๋ C++์std::unique_ptr
๊ณผ ๋น์ทํฉ๋๋ค. ์ฐจ์ด๋ผ๋ฉดBox
๋ ๋์ด ์๋์ ๋ณด์ฅํ๋ค๋ ์ ์ ๋๋ค.Deref
๋๋ถ์ ์ ์์ ์println!
๋ฌธ์ ์ฌ์ฉ๋*
๋ฅผ ๋นผ๋ ๋ฌธ์ ๊ฐ ์์ต๋๋ค.Box
๋ ์๋์ ๊ฒฝ์ฐ์ ์ ์ฉํฉ๋๋ค:- ํ์ ํฌ๊ธฐ๋ฅผ ์ปดํ์ผ ์์ ์ ์ ์ ์๋ ๊ฒฝ์ฐ.
- ์์ฃผ ํฐ ๋ฐ์ดํฐ์ ์์ ๊ถ์ ์ ๋ฌํ๊ณ ์ถ์ ๊ฒฝ์ฐ. ์คํ์ ์๋ ํฐ ๋ฐ์ดํฐ๋ฅผ ๋ณต์ฌํ๋ ๋์
Box
๋ฅผ ์ด์ฉํ์ฌ ๋ฐ์ดํฐ๋ ํ์ ์ ์ฅํ๊ณ ํฌ์ธํฐ๋ง ์ด๋ํ๋ฉด ๋ฉ๋๋ค.
์ฌ๊ท์๋ฃ ๊ตฌ์กฐ์์์ Box
์ฌ๊ท ๋ฐ์ดํฐ๋ ๋์ ํฌ๊ธฐ์ ๋ฐ์ดํฐ ํ์
์ Box
ํ์
์ ์ฌ์ฉํด์ผ ํฉ๋๋ค:
#[derive(Debug)] enum List<T> { Cons(T, Box<List<T>>), Nil, } fn main() { let list: List<i32> = List::Cons(1, Box::new(List::Cons(2, Box::new(List::Nil)))); println!("{list:?}"); }
-
๋ง์ผ
Box
๋ฅผ ์ฌ์ฉํ์ง ์๊ณList
์ ์ง์ List
๋ฅผ ํฌํจํ๋ ค๊ณ ์๋ํ๋ค๋ฉด, ์ปดํ์ผ๋ฌ๋ ๊ตฌ์กฐ์ฒด์ ๊ณ ์ ํฌ๊ธฐ๋ฅผ ๊ณ์ฐํ ์ ์์ต๋๋ค. ์ปดํ์ผ๋ฌ๊ฐ ๋ณด๊ธฐ์ ๋ฌดํ๋์ ํฌ๊ธฐ๋ก ๋ณด์ผ ๊ฒ์ ๋๋ค. -
Box
๋ ์ผ๋ฐ ํฌ์ธํฐ์ ํฌ๊ธฐ๊ฐ ๊ฐ๊ธฐ ๋๋ฌธ์ ํฌ๊ธฐ๋ฅผ ๊ณ์ฐํ๋ ๋ฐ ๋ฌธ์ ๊ฐ ์์ต๋๋ค. ๋ค๋ง ํ์ ์์นํList
์ ๋ค์ ์์๋ฅผ ๊ฐ๋ฆฌํฌ ๋ฟ์ ๋๋ค. -
List
์ ์์์Box
๋ฅผ ์ ๊ฑฐํ๋ฉด ์ด๋ค ์ปดํ์ผ๋ฌ ์๋ฌ๊ฐ ๋์ค๋์ง ๊ฐ์ด ์ดํด๋ณด์ธ์. โRecursive with indirectionโ๋ผ๋ ๋ฉ์์ง๋ฅผ ๋ณด๋ฉด, ๊ฐ์ ์ง์ ์ ์ฅํ๋ ๋์Box
๋ ๋น์ทํ ๋ค๋ฅธ ์ข ๋ฅ์ ์ฐธ์กฐ ํ์ ์ด ํ์ํ๋ค๋ ํํธ๋ฅผ ์ป์ ์ ์์ต๋๋ค.
๋์น(ํ์) ์ต์ ํ(Niche Optimization)
#[derive(Debug)] enum List<T> { Cons(T, Box<List<T>>), Nil, } fn main() { let list: List<i32> = List::Cons(1, Box::new(List::Cons(2, Box::new(List::Nil)))); println!("{list:?}"); }
Box
๋ ๋น์ด์์ ์ ์์ต๋๋ค. ๋ฐ๋ผ์ ํฌ์ธํฐ๋ ํญ์ ์ ํจํ๋ฉฐ null
์ด ์๋๋๋ค. ์ด๋ ์ปดํ์ผ๋ฌ๊ฐ ๋ฉ๋ชจ๋ฆฌ ๋ ์ด์์์ ์ต์ ํ ํ ์ ์๊ฒ ํด์ค๋๋ค:
Rc
Rc
๋ ์ฐธ์กฐ ์นด์ดํ
๊ณต์ ํฌ์ธํฐ์
๋๋ค. ์ฌ๋ฌ ์์น์์ ๋์ผํ ๋ฐ์ดํฐ๋ฅผ ์ฐธ์กฐํด์ผํ ๊ฒฝ์ฐ ์ฌ์ฉํฉ๋๋ค:
use std::rc::Rc; fn main() { let mut a = Rc::new(10); let mut b = Rc::clone(&a); println!("a: {a}"); println!("b: {b}"); }
- ๋ฉํฐ์ค๋ ๋ ํ๊ฒฝ์์ ์์
ํ๋ ๊ฒฝ์ฐ
Arc
์Mutex
๋ฅผ ์ฐธ์กฐํ์ธ์. - drop ๊ฐ๋ฅํ ์ํ ๊ตฌ์กฐ๋ฅผ ๋ง๋ค๊ธฐ ์ํด ๊ณต์ ํฌ์ธํฐ๋ฅผ
Weak
ํฌ์ธํฐ๋ก _๋ค์ด๊ทธ๋ ์ด๋_ํ ์๋ ์์ต๋๋ค.
Rc
๋ ์ฐธ์กฐ ์นด์ดํธ๋ฅผ ํตํด ์ฐธ์กฐ๊ฐ ์๋ ๋์์Rc
๊ฐ ๊ฐ๋ฆฌํค๊ณ ์๋ ๊ฐ์ด ๋ฉ๋ชจ๋ฆฌ์์ ํด์ ๋์ง ์์์ ๋ณด์ฅํฉ๋๋ค.- C++์
std::shared_ptr
์ ์ ์ฌํฉ๋๋ค. clone
์ ๋น์ฉ์ด ๊ฑฐ์ ๋ค์ง ์์ต๋๋ค. ๊ฐ์ ๊ณณ์ ๊ฐ๋ฆฌํค๋ ํฌ์ธํฐ๋ฅผ ํ๋ ๋ ๋ง๋ค๊ณ , ์ฐธ์กฐ ์นด์ดํธ๋ฅผ ๋๋ฆฝ๋๋ค. ํฌ์ธํฐ๊ฐ ๊ฐ๋ฆฌํค๋ ๊ฐ ์์ฒด๊ฐ ๋ณต์ (๊น์ ๋ณต์ )๋์ง๋ ์์ผ๋ฉฐ, ๊ทธ๋์ ์ฝ๋์์ ์ฑ๋ฅ ๋ฌธ์ ๊ฐ ์๋์ง ๊ฒํ ํ ๋ ์ผ๋ฐ์ ์ผ๋กRc
๋ฅผclone
ํ๋ ๊ฒ์ ๋ฌด์ํ ์ ์์ต๋๋ค.make_mut
๋ ์ค์ ๋ก ํ์ํ ๊ฒฝ์ฐ์ ๋ด๋ถ ๊ฐ์ ๋ณต์ ํ๊ณ (โclone-on-writeโ) ๊ฐ๋ณ ์ฐธ์กฐ๋ฅผ ๋ฐํํฉ๋๋ค.- ์ฐธ์กฐ ์นด์ดํธ๋ฅผ ํ์ธํ๋ ค๋ฉด
Rc::strong_count
๋ฅผ ์ฌ์ฉํ์ธ์. Rc
๋downgrade()
๋ก ๋ค์ด๊ทธ๋ ์ด๋ํ์ฌ ์ฝํ๊ฒ ์ฐธ์กฐ ์นด์ดํธ๋๋(weekly reference-counted) ๊ฐ์ฒด๊ฐ ๋ ์์ต๋๋ค. ๊ทธ๋ฌ๋ฉด ์ํ๊ตฌ์กฐ๋ผ ํ๋๋ผ๋ drop์ด ๊ฐ๋ฅํฉ๋๋ค. (์๋ง๋RefCell
์ ํจ๊ป ์ฌ์ฉํด์ผ ํ ๊ฒ์ ๋๋ค.)
Cell
๊ณผ RefCell
Cell
and RefCell
implement what Rust calls interior mutability: mutation of values in an immutable context.
Cell
์ ๊ฐ๋จํ ํ์
์ ๋ํด์ ์ฃผ๋ก ์ฌ์ฉ๋ฉ๋๋ค. ์๋ํ๋ฉด Cell
์ ์๋ ๊ฐ์ ์ฝ๊ฑฐ๋ ์ธ๋์๋ ๋ณต์ฌ ํน์ ์ด๋์ ํด์ผ๋ง ํ๊ธฐ ๋๋ฌธ์
๋๋ค. ๋ณต์กํ ํ์
์ด๋ผ๋ฉด RefCell
์ด ๋ ๋ณดํธ์ ์
๋๋ค. ์ด๋ฅผ ์ด์ฉํ๋ฉด ์ฐธ์กฐ๋ฅผ ํตํด ๊ฐ์ ์ฝ๊ฑฐ๋ ์ฐ๊ฒ ํด ์ฃผ๋ ๋์ , ๊ทธ ์ฐธ์กฐ๋ค์ด ์ฌ๋ฐ๋ฅธ์ง๋ฅผ ๋ฐํ์์ ์ฒดํฌํ๊ณ , ์ ๋๋ก ์ฌ์ฉ๋์ง ์์ ๊ฒฝ์ฐ ํจ๋์ ๋ฐ์์ํต๋๋ค.
use std::cell::RefCell; use std::rc::Rc; #[derive(Debug, Default)] struct Node { value: i64, children: Vec<Rc<RefCell<Node>>>, } impl Node { fn new(value: i64) -> Rc<RefCell<Node>> { Rc::new(RefCell::new(Node { value, ..Node::default() })) } fn sum(&self) -> i64 { self.value + self.children.iter().map(|c| c.borrow().sum()).sum::<i64>() } } fn main() { let root = Node::new(1); root.borrow_mut().children.push(Node::new(5)); let subtree = Node::new(10); subtree.borrow_mut().children.push(Node::new(11)); subtree.borrow_mut().children.push(Node::new(12)); root.borrow_mut().children.push(subtree); println!("graph: {root:#?}"); println!("graph sum: {}", root.borrow().sum()); }
- ์ด ์์ ์์
RefCell
๋์Cell
์ ์ผ์๋ค๋ฉด,Node
์ ์์ ๋ ธ๋๋ฅผ ์ถ๊ฐํ๊ธฐ ์ํด์,Node
๋ฅผRc
๋ฐ์ผ๋ก ์ด๋์ํจ ๋ค์, ์์ ๋ ธ๋๋ฅผ ์ถ๊ฐํ๊ณ , ๋ค์Rc
์์ผ๋ก ์ด๋์์ผ์ผ ํ์ ๊ฒ๋๋ค. ์ด๋ ๊ฒ ํด์ผ๋ง ํ๋ ์ด์ ๋ ์์ ๋๋ฌธ์ ๋๋ค. Cell ๋ด๋ถ์ ๊ทธ ๊ฐ์ด ์ค์ง ํ๋ ์กด์ฌํ๋ฉฐ, ๊ทธ ๊ฐ์ ๋ํ ์ฐธ์กฐ๊ฐ ์๋ค๋ ๊ฒ์ด ๋ณด์ฅ๋๊ธฐ ๋๋ฌธ์ ๋๋ค. ๋ฌผ๋ก ํธ๋ฆฌํ์ง๋ ์์ต๋๋ค. - ๋
ธ๋๋ฅผ ์ด๋ํ๊ฑฐ๋ ๋ณต์ฌํ์ง ์๊ณ ๊ทธ๋๋ก ์ฌ์ฉํ๊ธฐ ์ํด์๋
RefCell
๋ก ๊ฐ์ผ๋ค์borrow
๋borrow_mut
๋ฉ์๋๋ฅผ ์ด์ฉํด์ผ ํฉ๋๋ค. root
๋ฅผsubtree.children
์ ์ถ๊ฐํด์ ์ํ ์ฐธ์กฐ๊ฐ ์๊ธธ ์ ์์์ ๋ณด์ฌ์ฃผ์ธ์ (๊ทธ๋ํ๋ฅผ ์ถ๋ ฅํ์ง๋ ๋ง์ธ์!).self.value
๋ฅผ ์ฆ๊ฐ์ํค๋ ๋ฉ์๋์ธfn inc(&mut self)
๋ฅผ ์ถ๊ฐํ๊ณ ๊ทธ ๋ฉ์๋๋ฅผ ์์๋ ธ๋์์ ํธ์ถํ์ธ์. ๊ทธ๋ฌ๋ฉดthread 'main' panicked at 'already borrowed: BorrowMutError'
๋ฐํ์ ํจ๋์ด ๋ฐ์ํจ์ ๋ณด์ด์ธ์.
๋ชจ๋
impl
๋ธ๋ก์ ํด๋น ํ์
์ ํจ์๋ค์ ๋ํ ๋ค์์คํ์ด์ค๋ฅผ ์ ๊ณตํฉ๋๋ค.
๋ง์ฐฌ๊ฐ์ง๋ก, mod
๋ ํ์
๊ณผ ํจ์๋ค์ ๋ํด ๋ค์์คํ์ด์ค๋ฅผ ์ ๊ณตํฉ๋๋ค:
mod foo { pub fn do_something() { println!("In the foo module"); } } mod bar { pub fn do_something() { println!("In the bar module"); } } fn main() { foo::do_something(); bar::do_something(); }
- ํจํค์ง๋ ๊ธฐ๋ฅ์ ์ ๊ณตํ๋ฉฐ ํ๋์ ๋ํ
Cargo.toml
ํ์ผ์ ํฌํจํฉ๋๋ค. ํจํค์ง๋ฅผ ๊ตฌ์ฑํ๋ ํฌ๋ ์ดํธ๋ค์ ๋น๋ํ๋ ๋ฐฉ๋ฒ์ด ์ด ํ์ผ์ ๊ธฐ์ ๋ฉ๋๋ค. - ํฌ๋ ์ดํธ๋ ๋ชจ๋์ ํธ๋ฆฌ์ ๋๋ค. ๋ฐ์ด๋๋ฆฌ ํฌ๋ ์ดํธ๋ ์คํํ์ผ๋ก ๋น๋๋๊ณ , ๋ผ์ด๋ธ๋ฌ๋ฆฌ ํฌ๋ ์ดํธ๋ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ก ๋น๋๋ฉ๋๋ค.
- ๋ชจ๋์ ์ฝ๋๋ฅผ ์กฐ์งํํ๊ณ ์ค์ฝํ๋ฅผ ์ ์ํ๋ ๋จ์์ ๋๋ค.
๊ฐ์์ฑ
๋ชจ๋์ ํ์ ์ด๋ ํจ์๋ ๊ธฐ๋ณธ์ ์ผ๋ก ๋ฐ๊นฅ์ ๋ ธ์ถ๋์ง ์์ต๋๋ค:
- ๋ฐ๋ผ์ ๋ชจ๋์ ์ธ๋ถ ๊ตฌํ ๋ด์ฉ์ด ๊ฐ์ถฐ์ง๋๋ค.
- ๋ถ๋ชจ์ ์ด์ ํญ๋ชฉ์ ์ธ์ ๋ ์ ๊ทผ ๊ฐ๋ฅํฉ๋๋ค.
- ์ฆ, ๋ชจ๋
foo
์์ ์ ๊ทผ ๊ฐ๋ฅํ ํญ๋ชฉ์ด๋ผ๋ฉดfoo
์๋์ ๋ชจ๋ ๋ชจ๋์์ ์ ๊ทผ๊ฐ๋ฅํฉ๋๋ค.
mod outer { fn private() { println!("outer::private"); } pub fn public() { println!("outer::public"); } mod inner { fn private() { println!("outer::inner::private"); } pub fn public() { println!("outer::inner::public"); super::private(); } } } fn main() { outer::public(); }
pub
ํค์๋๋ ๋ชจ๋์๋ ์ฌ์ฉํ ์ ์์ต๋๋ค.
๋ํ, ๊ณ ๊ธ ๊ธฐ๋ฅ์ผ๋ก pub(...)
์ง์ ์๋ฅผ ์ฌ์ฉํ์ฌ ๊ณต๊ฐ ๋ฒ์๋ฅผ ์ ํํ ์ ์์ต๋๋ค.
- ๊ณต์ ๋ฌธ์๋ฅผ ์ฐธ๊ณ ํ์ธ์.
pub(crate)
๋ก ๊ฐ์์ฑ์ ์ง์ ํ๋ ๊ฒ์ด ์์ฃผ ์ฐ์ ๋๋ค.- ์์ฃผ ์ฐ์ด์ง ์์ง๋ง ํน์ ๊ฒฝ๋ก์ ๋ํด์๋ง ๊ฐ์์ฑ์ ๋ถ์ฌํ ์ ์์ต๋๋ค.
- ์ด๋ค ๊ฒฝ์ฐ์ด๋ ๊ฐ์์ฑ์ด ๋ถ์ฌ๋๋ฉด ํด๋น ๋ชจ๋์ ํฌํจํ์ฌ ํ์์ ๋ชจ๋ ๋ชจ๋์ด ์ ์ฉ๋ฐ์ต๋๋ค.
๊ฒฝ๋ก
๊ฒฝ๋ก๋ ์๋์ ๊ฐ์ด ๊ตฌ๋ถํฉ๋๋ค:
-
์๋ ๊ฒฝ๋ก:
foo
๋๋self::foo
๋ ํ์ฌ ๋ชจ๋ ๋ด๋ถ์foo
๋ฅผ ๊ฐ๋ฆฌํต๋๋ค,super::foo
๋ ๋ถ๋ชจ ๋ชจ๋์foo
๋ฅผ ๊ฐ๋ฆฌํต๋๋ค.
-
์ ๋ ๊ฒฝ๋ก:
crate::foo
๋ ํ์ฌ ํฌ๋ ์ดํธ ๋ฃจํธ์foo
๋ฅผ ๊ฐ๋ฆฌํต๋๋ค,bar::foo
๋bar
ํฌ๋ ์ดํธ์foo
๋ฅผ ๊ฐ๋ฆฌํต๋๋ค.
๋ชจ๋์ use
๋ฅผ ์ฌ์ฉํ์ฌ ๋ค๋ฅธ ๋ชจ๋์ ์ฌ๋ณผ์ ๋ด ์ค์ฝํ๋ก ๊ฐ์ ธ์ฌ ์ ์์ต๋๋ค. ์ผ๋ฐ์ ์ผ๋ก ๊ฐ ๋ชจ๋์ ์๋จ์ ๋ค์๊ณผ ๊ฐ์ ๋ด์ฉ์ด ์ต๋๋ค:
use std::collections::HashSet; use std::mem::transmute;
ํ์ผ์์คํ ๊ณ์ธต
๋ชจ๋์ ๋ด์ฉ์ ๊ธฐ์ ํ์ง ์์ผ๋ฉด, ๋ฌ์คํธ๋ ๋ค๋ฅธ ํ์ผ์์ ๊ทธ ๋ด์ฉ์ ์ฝ์ต๋๋ค:
mod garden;
์ ์ฝ๋๋ ๋ฌ์คํธ๋ก ํ์ฌ๊ธ garden
๋ชจ๋์ ๋ด์ฉ์ src/garden.rs
์์ ์ฐพ๋๋ก ํฉ๋๋ค. ๋น์ทํ๊ฒ, garden::vegetables
๋ชจ๋์ src/garden/vegetables.rs
์์ ์ฐพ์ต๋๋ค.
crate(ํฌ๋ ์ดํธ)
์ ๋ฃจํธ๋ ์๋ ๊ฒฝ๋ก ์
๋๋ค:
src/lib.rs
(๋ผ์ด๋ธ๋ฌ๋ฆฌ ํฌ๋ ์ดํธ)src/main.rs
(๋ฐ์ด๋๋ฆฌ ํฌ๋ ์ดํธ)
๋ชจ๋๋ โ๋ด๋ถ ๋ฌธ์ ์ฃผ์โ์ ์ฌ์ฉํ์ฌ ๋ฌธ์ํํ ์ ์์ต๋๋ค. ์ด๋ฌํ ๋ชจ๋์ ๋ชจ๋์ด ํฌํจ๋ ํญ๋ชฉ(์ด ๊ฒฝ์ฐ์๋ ๋ชจ๋)์ ๋ฌธ์ํํฉ๋๋ค.
//! This module implements the garden, including a highly performant germination //! implementation. // Re-export types from this module. pub use seeds::SeedPacket; pub use garden::Garden; /// Sow the given seed packets. pub fn sow(seeds: Vec<SeedPacket>) { todo!() } /// Harvest the produce in the garden that is ready. pub fn harvest(garden: &mut Garden) { todo!() }
-
module/mod.rs
๋ฅผmodule.rs
๋ก ๋ฐ๊พผ๋ค ํ๋๋ผ๋ Rust 2018์์๋ ํ์ ๋ชจ๋์ ์ฌ์ฉํ ์ ์์ต๋๋ค. -
filename.rs
๋ฅผfilename/mod.rs
๋์ ์ฌ์ฉํ ์ ์๋๋ก ํ๋ ์ฃผ๋ ์ด์ ๋,mod.rs
๋ผ๋ ์ด๋ฆ์ ๊ฐ์ง ํ์ผ์ด ๋ง์ ๊ฒฝ์ฐ IDE์์ ์ด๋ค์ ์๋ก ๊ตฌ๋ณํ๋๊ฒ ํ๋ค๊ธฐ ๋๋ฌธ์ ๋๋ค. -
ํด๋๋ฅผ ์ด์ฉํด์ ๋ ๊น์ ๊ณ์ธต ๊ตฌ์กฐ๋ฅผ ๊ตฌํํ ์ ์์ต๋๋ค. ์ฌ์ง์ด๋ ๋ฉ์ธ ๋ชจ๋์ด ํ์ผ์ด๋๋ผ๋์:
src/ โโโ main.rs โโโ top_module.rs โโโ top_module/ โโโ sub_module.rs
-
๋ฌ์คํธ๊ฐ ์ด๋์ ๋ชจ๋๋ค์ ์ฐพ์์ง๋ ์ปดํ์ผ๋ฌ ๋๋ ํฐ๋ธ๋ก ๋ณ๊ฒฝ ๊ฐ๋ฅํฉ๋๋ค:
#[path = "some/path.rs"] mod some_module;
์ด๋ Go์ธ์ด ์์์ฒ๋ผ ์ด๋ค ๋ชจ๋์ ํ ์คํธ๋ฅผ
some_module_test.rs
๊ฐ์ ํ์ผ์ ๋๊ณ ์ถ์ ๊ฒฝ์ฐ์ ์ ์ฉํฉ๋๋ค.
2์ผ์ฐจ ์คํ ์ฐ์ต๋ฌธ์
์ด๋ฒ ์ฐ์ต๋ฌธ์ ๋ ๋ฌธ์์ด๊ณผ ๋ฐ๋ณต์์ ์ด์ ์ ๋ง์ถ ๊ฒ์ ๋๋ค.
After looking at the exercises, you can look at the solutions provided.
๋ฐ๋ณต์์ ์์ ๊ถ
๋ฌ์คํธ์ ์์ ๊ถ ๋ชจ๋ธ์ ๋ง์ API์ ๋ฐ์์ด ๋์ด ์์ต๋๋ค. ์๋ฅผ๋ค์ด Iterator
์ IntoIterator
๊ฐ์ ํธ๋ ์์ด ์์ต๋๋ค.
Iterator
ํธ๋ ์์ ํ์
์ ๋ํ ํ๋(๋ฉ์๋)๋ฅผ ์ค๋ช
ํ๋ค๋ ์ ์์ ์ธํฐํ์ด์ค์ ์ ์ฌํฉ๋๋ค. Iterator
๋ ๋จ์ํ None
์ด ๋์ฌ๋๊น์ง next
๋ฅผ ํธ์ถํ๋ ๊ฒ์ด ๊ฐ๋ฅํ๋ค๋ ๊ฒ์ ๋ํ๋ด๋ ํธ๋ ์์
๋๋ค:
#![allow(unused)] fn main() { pub trait Iterator { type Item; fn next(&mut self) -> Option<Self::Item>; } }
Iterator
ํธ๋ ์์ ์ด๋ ๊ฒ ์ฌ์ฉํฉ๋๋ค:
fn main() { let v: Vec<i8> = vec![10, 20, 30]; let mut iter = v.iter(); println!("v[0]: {:?}", iter.next()); println!("v[1]: {:?}", iter.next()); println!("v[2]: {:?}", iter.next()); println!("No more items: {:?}", iter.next()); }
๋ฐ๋ณต์๊ฐ ๋ฐํํ๋ ๊ฐ๋ค์ ํ์ ์ด ๋ญ๊น์? ์ฌ๊ธฐ์ ๋ต์ ํ ์คํธ ํด ๋ณด์ธ์:
fn main() { let v: Vec<i8> = vec![10, 20, 30]; let mut iter = v.iter(); let v0: Option<..> = iter.next(); println!("v0: {v0:?}"); }
์ ์ด๋ฐ ํ์ ์ด ์ฌ์ฉ๋๋ ๊ฒ์ผ๊น์?
IntoIterator
Iterator
ํธ๋ ์์ ์์ฑ๋ ๋ฐ๋ณต์๋ฅผ _์ฌ์ฉ_ํ๋ ๋ฐฉ๋ฒ์ ์๋ ค์ค๋๋ค. ๋ฐ๋ฉด IntoIterator
ํธ๋ ์์ ๋ฐ๋ณต์๋ฅผ _์์ฑ_ํ๋ ๋ฐฉ๋ฒ์ ์๋ ค์ค๋๋ค:
#![allow(unused)] fn main() { pub trait IntoIterator { type Item; type IntoIter: Iterator<Item = Self::Item>; fn into_iter(self) -> Self::IntoIter; } }
IntoIterator
์ ๋ชจ๋ ๊ตฌํ์ ๋ฐ๋์ ๋ค์์ ๋ ํ์
์ ์ ์ธํด์ผํฉ๋๋ค:
Item
:i8
๊ณผ ๊ฐ์ด ๋ฐ๋ณต๋๋ ๊ฐ์ ํ์ ,IntoIter
:into_iter
๋ฉ์๋์์ ๋ฐํ๋๋Iterator
ํ์ .
IntoIter
์๋ Item
์ด ์ฐ๊ฒฐ๋์ด ์์์ ์ฃผ๋ชฉํ์ธ์. IntoIter
๋ฐ๋ณต์๋ Item
ํ์
์ ๋ฐ์ดํฐ๋ฅผ ๊ฐ๋ฆฌ์ผ์ผ ํฉ๋๋ค. ์ฆ, ๋ฐ๋ณต์๋ Option<Item>
์ ๋ฆฌํดํฉ๋๋ค
์ด์ ๊ณผ ๋ง์ฐฌ๊ฐ์ง๋ก, ๋ฐ๋ณต์๊ฐ ๋ฐํํ๋ ํ์ ์ ๋ฌด์์ ๋๊น?
fn main() { let v: Vec<String> = vec![String::from("foo"), String::from("bar")]; let mut iter = v.into_iter(); let v0: Option<..> = iter.next(); println!("v0: {v0:?}"); }
๋ฐฐ์ด๊ณผ for
๋ฐ๋ณต๋ฌธ
์, ์ด์ ์ฐ๋ฆฌ๋ Iterator
์ IntoIterator
๋ฅผ ์์์ผ๋ฏ๋ก for
๋ฃจํ๋ฅผ ๋ง๋ค ์ ์์ต๋๋ค. for
๋ฃจํ๋ into_iter()
๋ฅผ ํธ์ถํ์ฌ ๋ฐ๋ณต์๋ฅผ ๋ง๋ ๋ค์ ๊ทธ ๋ฐ๋ณต์๋ฅผ ์ด์ฉํ์ฌ ์์๋ค์ ๋ฐ๋ณตํด์ ์ ๊ทผํฉ๋๋ค:
fn main() { let v: Vec<String> = vec![String::from("foo"), String::from("bar")]; for word in &v { println!("word: {word}"); } for word in v { println!("word: {word}"); } }
๋งค ๋ฃจํ์์ word
์ ํ์
์ ๋ฌด์์
๋๊น?
์ ์ฝ๋์์ ์คํ ํด ๋ณธ ํ, ๋ค์ ๋ฌธ์๋ฅผ ์ฐธ์กฐํด์ ๋ต๋ณ์ ํ์ธํ์๊ธฐ ๋ฐ๋๋๋ค:impl IntoIterator for &Vec<T>
, impl IntoIterator for Vec<T>
๋ฌธ์์ด๊ณผ ๋ฐ๋ณต์
์ด๋ฒ ํ๋ จ์ ์น ์๋ฒ์ ๋ผ์ฐํ ์ปดํฌ๋ํธ๋ฅผ ๊ตฌํํฉ๋๋ค. ์๋ฒ๋ ์์ฒญ ๊ฒฝ๋ก(request path) ๋ฅผ ์ฒ๋ฆฌํ ์ ์๋ ์ฌ๋ฌ ๊ฐ์ ๊ฒฝ๋ก ์ ๋์ฌ(path prefix) ๋ก ๊ตฌ์ฑ๋ฉ๋๋ค. ๊ฒฝ๋ก ์ ๋์ฌ๋ ์์ผ๋์นด๋๋ฌธ์๋ฅผ ํฌํจํ ์ ์์ต๋๋ค. ์๋ ๋จ์ ํ ์คํธ ์ฝ๋๋ฅผ ์ฐธ์กฐํ์ธ์.
์๋ ์ฝ๋๋ฅผ https://play.rust-lang.org/์ ๋ณต์ฌํ๊ณ ํ
์คํธ๋ฅผ ํต๊ณผํด ๋ณด์๊ธฐ ๋ฐ๋๋๋ค. ์ค๊ฐ ๊ฒฐ๊ณผ๊ฐ์ Vec
์ ํ ๋นํ์ง ์๋๋ก ์ฃผ์ ํ์๊ธฐ ๋ฐ๋๋๋ค:
#![allow(unused)] fn main() { // TODO: remove this when you're done with your implementation. #![allow(unused_variables, dead_code)] pub fn prefix_matches(prefix: &str, request_path: &str) -> bool { unimplemented!() } #[test] fn test_matches_without_wildcard() { assert!(prefix_matches("/v1/publishers", "/v1/publishers")); assert!(prefix_matches("/v1/publishers", "/v1/publishers/abc-123")); assert!(prefix_matches("/v1/publishers", "/v1/publishers/abc/books")); assert!(!prefix_matches("/v1/publishers", "/v1")); assert!(!prefix_matches("/v1/publishers", "/v1/publishersBooks")); assert!(!prefix_matches("/v1/publishers", "/v1/parent/publishers")); } #[test] fn test_matches_with_wildcard() { assert!(prefix_matches( "/v1/publishers/*/books", "/v1/publishers/foo/books" )); assert!(prefix_matches( "/v1/publishers/*/books", "/v1/publishers/bar/books" )); assert!(prefix_matches( "/v1/publishers/*/books", "/v1/publishers/foo/books/book1" )); assert!(!prefix_matches("/v1/publishers/*/books", "/v1/publishers")); assert!(!prefix_matches( "/v1/publishers/*/books", "/v1/publishers/foo/booksByAuthor" )); } }
3์ผ์ฐจ ๊ฐ์
์ค๋์ ๋ช ๊ฐ์ง ๊ณ ๊ธ ์ฃผ์ ๋ฅผ ๋ค๋ฃน๋๋ค:
-
ํธ๋ ์: ํธ๋ ์ ์์(derive), ๋ํดํธ ๋ฉ์๋, ํ์ค ๋ผ์ด๋ธ๋ฌ์ ์๋ ์ค์ํ ํธ๋ ์๋ค.
-
์ ๋ค๋ฆญ: ์ ๋ค๋ฆญ ๋ฐ์ดํฐ ํ์ , ์ ๋ค๋ฆญ ๋ฉ์๋, ๋จํํ(monomorphization), ํธ๋ ์ ๊ฐ์ฒด.
-
์ค๋ฅ์ฒ๋ฆฌ(์๋ฌ ํธ๋ค๋ง): ํจ๋,
Result
,?
์ฐ์ฐ์. -
ํ ์คํธ: ๋จ์ ํ ์คํธ, ๋ฌธ์ ํ ์คํธ ๋ฐ ํตํฉ ํ ์คํธ.
-
์์ ํ์ง ์์ ๋ฌ์คํธ: ์์(raw) ํฌ์ธํฐ, ์ ์ ๋ณ์, ์์ ํ์ง ์์ ํจ์, ์ธ๋ถ ํจ์.
์ ๋ค๋ฆญ
๋ฌ์คํธ๋ ์ ๋ค๋ฆญ์ ์ง์ํฉ๋๋ค. ์ด๋ฅผ ์ด์ฉํ๋ฉด ์๊ณ ๋ฆฌ์ฆ(์ ๋ ฌ๊ณผ ๊ฐ์)์ด๋ ๋ฐ์ดํฐ ๊ตฌ์กฐ(์ด์ง ํธ๋ฆฌ ๊ฐ์)๋ฅผ ์ถ์ํ ํ์ฌ ํน์ ํ์ ์ ์์กดํ์ง ์๋๋ก ํ ์ ์์ต๋๋ค.
์ ๋ค๋ฆญ ๋ฐ์ดํฐ ํ์
์ ๋ค๋ฆญ์ ์ฌ์ฉํ์ฌ ํ๋์ ํ์ ์ ์ถ์ํ ํ ์ ์์ต๋๋ค:
#[derive(Debug)] struct Point<T> { x: T, y: T, } fn main() { let integer = Point { x: 5, y: 10 }; let float = Point { x: 1.0, y: 4.0 }; println!("{integer:?} and {float:?}"); }
-
์ ๋ณ์
let p = Point { x: 5, y: 10.0 };
๋ฅผ ์ ์ธํด ๋ณด์ธ์. -
Point
๊ฐ ์๋ก ๋ค๋ฅธ ํ์ ์ ๊ฐ๋ค์ผ๋ก ์ด๋ฃจ์ด์ ธ๋ ์ปดํ์ผ ๋๋๋ก ์ฝ๋๋ฅผ ์์ ํด ๋ณด์ธ์.
์ ๋ค๋ฆญ ๋ฉ์๋
impl
๋ธ๋ก์์๋ ์ ๋ค๋ฆญ ํ์
์ ์ ์ธํ ์ ์์ต๋๋ค:
#[derive(Debug)] struct Point<T>(T, T); impl<T> Point<T> { fn x(&self) -> &T { &self.0 // + 10 } // fn set_x(&mut self, x: T) } fn main() { let p = Point(5, 10); println!("p.x = {}", p.x()); }
- ์ง๋ฌธ:
impl<T> Point<T> {}
์์T
๊ฐ ์ ๋ ๋ฒ ์ฌ์ฉ๋ฉ๋๊น?- ์ ๋ค๋ฆญ ํ์ ์ ๋ํ ์ ๋ค๋ฆญ ๊ตฌํ ์ด๊ธฐ ๋๋ฌธ์ ๋๋ค. ์ด ๋ ์ ๋ค๋ฆญ์ ์๋ก ๋ ๋ฆฝ์ ์ ๋๋ค.
- ์ด๋ ์์์ ๋ชจ๋
T
์ ๋ํด์ ์ด ๋ฉ์๋๋ค์ด ์ ์๋๋ค๋ ๊ฒ์ ์๋ฏธํฉ๋๋ค. impl Point<u32> { .. }
์ ๊ฐ์ด ์์ฑํ๋ ๊ฒ๋ ๊ฐ๋ฅํฉ๋๋ค.Point
๋ ์ฌ์ ํ ์ ๋ค๋ฆญ์ด๋ฉฐPoint<f64>
๋ฅผ ์ฌ์ฉํ ์๋ ์์ง๋ง ์ด ๋ธ๋ก์ ๋ฉ์๋๋Point<u32>
๋ง ์ธ ์ ์์ต๋๋ค.
๋จํํ
์ ๋ค๋ฆญ ์ฝ๋๋ ํธ์ถ๋ถ์์ ๋น ์ ๋ค๋ฆญ ์ฝ๋๋ก ์ ํ๋ฉ๋๋ค:
fn main() { let integer = Some(5); let float = Some(5.0); }
์ ์ฝ๋๋ ์๋์ ๊ฐ์ด ๋์ํฉ๋๋ค
enum Option_i32 { Some(i32), None, } enum Option_f64 { Some(f64), None, } fn main() { let integer = Option_i32::Some(5); let float = Option_f64::Some(5.0); }
์ด๊ฒ์ด ๋ฐ๋ก ๋น์ฉ์ด ๋ค์ง ์๋ (zero-cost) ์ถ์ํ ์ ๋๋ค: ๋ฌ์คํธ์ ์ ๋ค๋ฆญ์ ์ถ์ํ๋ฅผ ๊ฑฐ์น์ง ์๊ณ ์ง์ ๊ตฌ์ฒด์ ์ธ ํ์ ์ ์จ์ ์ฝ๋ฉํ ๊ฒ๊ณผ ์ ํํ ๋์ผํ ๊ฒฐ๊ณผ๋ฅผ ๋ณด์ฌ์ค๋๋ค.
ํธ๋ ์(Trait)
ํธ๋ ์์ ํ์ ์ ์ถ์ํ ํ๋๋ฐ ์ฌ์ฉ๋ฉ๋๋ค. ์ธํฐํ์ด์ค์ ๋น์ทํฉ๋๋ค:
trait Pet { fn name(&self) -> String; } struct Dog { name: String, } struct Cat; impl Pet for Dog { fn name(&self) -> String { self.name.clone() } } impl Pet for Cat { fn name(&self) -> String { String::from("The cat") // No name, cats won't respond to it anyway. } } fn greet<P: Pet>(pet: &P) { println!("Who's a cutie? {} is!", pet.name()); } fn main() { let fido = Dog { name: "Fido".into() }; greet(&fido); let captain_floof = Cat; greet(&captain_floof); }
ํธ๋ ์ ๊ฐ์ฒด
ํธ๋ ์ ๊ฐ์ฒด๋ ํ์ ์ด ๋ค๋ฅธ ๊ฐ(์๋ฅผ ๋ค์ด ์ปฌ๋ ์ ์ ์ํ ๊ฐ ๊ฐ)๋ค์ ๊ฐ์ง ์ ์์ต๋๋ค:
trait Pet { fn name(&self) -> String; } struct Dog { name: String, } struct Cat; impl Pet for Dog { fn name(&self) -> String { self.name.clone() } } impl Pet for Cat { fn name(&self) -> String { String::from("The cat") // No name, cats won't respond to it anyway. } } fn main() { let pets: Vec<Box<dyn Pet>> = vec![ Box::new(Cat), Box::new(Dog { name: String::from("Fido") }), ]; for pet in pets { println!("Hello {}!", pet.name()); } }
pets
๋ฅผ ํ ๋นํ ์ดํ์ ๋ฉ๋ชจ๋ฆฌ ๋ ์ด์์:
- ์ฌ๋ฌ ํ์
์ด ๊ฐ์ ํธ๋ ์์ ๊ตฌํํ๋๋ผ๋ ๊ทธ ํฌ๊ธฐ๋ ์๋ก ๋ค๋ฅผ ์ ์์ต๋๋ค. ๊ทธ๋์
Vec<Greeet>
๊ฐ์ ๊ฒ์ ๋ถ๊ฐ๋ฅํฉ๋๋ค. dyn Pet
์ด๋ผ๊ณ ํ๋ฉด์ด ํ์ ์ ํฌ๊ธฐ๋ ๋์ ์ด๋ฉฐPet
์ ๊ตฌํํ๊ณ ์๋ค๊ณ ์ปดํ์ผ๋ฌ์๊ฒ ์๋ ค์ฃผ๋ ๊ฒ์ ๋๋ค.- ์์ ์์
pets
๋Pet
์ ๊ตฌํํ๋ ๊ฐ์ฒด๋ค์ _Fat ํฌ์ธํฐ_๋ฅผ ๋ด๊ณ ์์ต๋๋ค. Fat ํฌ์ธํฐ๋ ์ค์ ๊ฐ์ฒด์ ๋ํ ํฌ์ธํฐ์ ๊ทธ ๊ฐ์ฒด๊ฐPet
์ ๊ตฌํํ๊ณ ์๋ ๊ฐ์ ํจ์ ํ ์ด๋ธ์ ๋ํ ํฌ์ธํฐ๋ฅผ ๊ฐ์ง๋๋ค. - ์๋ ์ฝ๋์ ๊ฒฐ๊ณผ์ ๋น๊ตํด๋ณด์ธ์:
println!("{} {}", std::mem::size_of::<Dog>(), std::mem::size_of::<Cat>()); println!("{} {}", std::mem::size_of::<&Dog>(), std::mem::size_of::<&Cat>()); println!("{}", std::mem::size_of::<&dyn Pet>()); println!("{}", std::mem::size_of::<Box<dyn Pet>>());
ํธ๋ ์ ์์ํ๊ธฐ
๋ฌ์คํธ์ derive ๋งคํฌ๋ก๋ ๋ฐ์ดํฐ ๊ตฌ์กฐ์ฒด๊ฐ ํน์ ํธ๋ ์์ ๊ตฌํํ๋ ์ฝ๋๋ฅผ ์๋์ผ๋ก ๋ง๋ค์ด ์ค๋๋ค.
์ปดํ์ผ๋ฌ๊ฐ ์ฌ๋ฌ๊ฐ์ง ํธ๋ ์์ ์์(derive)ํ๋๋ก ํ ์ ์์ต๋๋ค. ์ด ๊ฒฝ์ฐ ์ปดํ์ผ๋ฌ๊ฐ ํธ๋ ์์ ์๋์ผ๋ก ๊ตฌํํฉ๋๋ค:
#[derive(Debug, Clone, PartialEq, Eq, Default)] struct Player { name: String, strength: u8, hit_points: u8, } fn main() { let p1 = Player::default(); let p2 = p1.clone(); println!("Is {:?}\nequal to {:?}?\nThe answer is {}!", &p1, &p2, if p1 == p2 { "yes" } else { "no" }); }
๊ธฐ๋ณธ ๋ฉ์๋
ํธ๋ ์์ ๋ํดํธ ๋ฉ์๋์์ ๋ค๋ฅธ(๊ตฌํ๋์ง ์์) ๋ฉ์๋๋ฅผ ์ด์ฉํ ์ ์์ต๋๋ค:
trait Equals { fn equals(&self, other: &Self) -> bool; fn not_equals(&self, other: &Self) -> bool { !self.equals(other) } } #[derive(Debug)] struct Centimeter(i16); impl Equals for Centimeter { fn equals(&self, other: &Centimeter) -> bool { self.0 == other.0 } } fn main() { let a = Centimeter(10); let b = Centimeter(20); println!("{a:?} equals {b:?}: {}", a.equals(&b)); println!("{a:?} not_equals {b:?}: {}", a.not_equals(&b)); }
-
ํธ๋ ์์, ๊ทธ ํธ๋ ์์ด ์ ์๋ ์์น์์ ์ง์ ๊ตฌํ๋๋ ๋ํดํธ ๋ฉ์๋์, ๊ฑฐ๊ธฐ์์๋ ์ ์ธ๋ง ์กด์ฌํ๊ณ ๊ทธ ํธ๋ ์์ ๊ตฌํํ๋ ํ์ ์์ ๊ตฌํํด์ผ ํ๋, ๋ ์ข ๋ฅ์ ๋ฉ์๋๋ฅผ ๊ฐ์ง ์ ์์ต๋๋ค. ๋ํดํธ ๋ฉ์๋๋ฅผ ๊ตฌํํ ๋์๋, ๋ ์ข ๋ฅ์ ๋ฉ์๋ ๋ชจ๋ ์ฌ์ฉ(ํธ์ถ)ํ ์ ์์ต๋๋ค.
-
not_equal
๋ฉ์๋๋ฅผ ์๋ก์ด ํธ๋ ์์ธNotEqual
๋ก ์ด๋ํฉ๋๋ค. -
NotEqual
์Equal
์ ์ํผ ํธ๋ ์์ผ๋ก ๋ง๋ญ๋๋ค.trait NotEquals: Equals { fn not_equals(&self, other: &Self) -> bool { !self.equals(other) } }
-
Equal
์NotEqual
์ ํฌ๊ด์ ๊ตฌํ์ ์ ๊ณตํฉ๋๋ค.trait NotEquals { fn not_equals(&self, other: &Self) -> bool; } impl<T> NotEquals for T where T: Equals { fn not_equals(&self, other: &Self) -> bool { !self.equals(other) } }
- ํฌ๊ด์ ๊ตฌํ์ ์ฌ์ฉํ๋ฉด ๋ ์ด์
NotEqual
์ดEqual
์ ์ํผ ํธ๋ ์์ผ๋ก ํ์ํ์ง ์์ต๋๋ค.
- ํฌ๊ด์ ๊ตฌํ์ ์ฌ์ฉํ๋ฉด ๋ ์ด์
์ ๋ค๋ฆญ ํ์ ์ ํ(ํธ๋ ์ ๊ฒฝ๊ณ)
์ ๋ค๋ฆญ์ ์ด์ฉํ๋ค ๋ณด๋ฉด ํ์ ์ด ์ด๋ค ํธ๋ ์์ ๊ตฌํํ๊ณ ์์ด์ผ ํ๋ ๊ฒฝ์ฐ๊ฐ ์์ต๋๋ค. ๊ทธ๋์ผ ๊ทธ ํธ๋ ์์ ๋ฉ์๋๋ฅผ ํธ์ถํ ์ ์๊ธฐ ๋๋ฌธ์ ๋๋ค.
T: Trait
ํน์ impl Trait
๋ฅผ ์ฌ์ฉํ๋ฉด ๋ฉ๋๋ค:
fn duplicate<T: Clone>(a: T) -> (T, T) { (a.clone(), a.clone()) } // Syntactic sugar for: // fn add_42_millions<T: Into<i32>>(x: T) -> i32 { fn add_42_millions(x: impl Into<i32>) -> i32 { x.into() + 42_000_000 } // struct NotClonable; fn main() { let foo = String::from("foo"); let pair = duplicate(foo); println!("{pair:?}"); let many = add_42_millions(42_i8); println!("{many}"); let many_more = add_42_millions(10_000_000); println!("{many_more}"); }
where
๋ฌธ๋ฒ์ ์ฌ์ฉํ ์๋ ์์ต๋๋ค. ์๊ฐ์๋ค๋ ์ฝ๋๋ฅผ ์ฝ๋ค๊ฐ ๊ทธ ๋ฌธ๋ฒ์ ๋ง์ฃผํ ์ ์์ต๋๋ค.
fn duplicate<T>(a: T) -> (T, T)
where
T: Clone,
{
(a.clone(), a.clone())
}
- ์ด๋ฅผ ์ด์ฉํ๋ฉด ํ์ ํ๋ผ๋ฉํฐ๊ฐ ๋ง์ ๊ฒฝ์ฐ ํจ์ ์๊ทธ๋์ฒ๋ฅผ ๊ฐ๊ฒฐํ๊ฒ ์ ๋ฆฌํ๋ ๋ฐ ๋์์ด ๋ฉ๋๋ค.
- ์ข ๋ ๊ฐ๋ ฅํ ์ถ๊ฐ ๊ธฐ๋ฅ๋ ์ ๊ณตํฉ๋๋ค.
:
์ผ์ชฝ์ ์์์ ํ์ (์๋ฅผ ๋ค์ดOption<T>
)์ ์ฌ์ฉํ ์ ์์ต๋๋ค.
ํธ๋ ์ ๊ตฌํํ๊ธฐ(impl Trait
)
ํธ๋ ์ ๋ฐ์ด๋์ ์ ์ฌํ๊ฒ impl Trait
๋ฌธ๋ฒ์ ํจ์์ ์ธ์์ ๋ฐํ๊ฐ์๋ ์ ์ฉ ๊ฐ๋ฅํฉ๋๋ค:
use std::fmt::Display; fn get_x(name: impl Display) -> impl Display { format!("Hello {name}") } fn main() { let x = get_x("foo"); println!("{x}"); }
impl Trait
๋ฅผ ์ด์ฉํ๋ฉด ์ด๋ฆ์ด ์๋ ํ์ ์ ๋ค๋ฃฐ ์ ์์ต๋๋ค.
impl Trait
๋ ์ด๋์ ์ฌ์ฉ๋์๋๋์ ๋ฐ๋ผ ์๋ฏธ๊ฐ ์กฐ๊ธ์ฉ ๋ค๋ฆ
๋๋ค.
-
ํจ์ ์ธ์์ ํ์ ์ผ๋ก ์ฌ์ฉ๋์์ ๊ฒฝ์ฐ์๋
impl Trait
๋ ํธ๋ ์ ๊ฒฝ๊ณ๊ฐ ์๋ ์ต๋ช ์ ์ ๋ค๋ฆญ ํ์ ์ ์๋ฏธํฉ๋๋ค. -
๋ฆฌํด ํ์ ์ผ๋ก ์ฌ์ฉ๋์์ ๊ฒฝ์ฐ์๋, ๊ทธ ํธ๋ ์์ ๊ตฌํํ๋ ๊ตฌ์ฒด์ ์ธ ํ์ ์ธ๋ฐ, ํ์ ์ด๋ฆ์ ํ๋ก๊ทธ๋๋จธ๊ฐ ์ง์ง ์์๋ค๋ ๊ฒ์ ์๋ฏธํฉ๋๋ค. ์ด๋ ๊ทธ ๊ตฌ์ฒด์ ์ธ ํ์ ์ด๋ฆ์ API๋ก ๊ณต๊ฐํ๊ณ ์ถ์ง ์์ ๊ฒฝ์ฐ์ ์ ์ฉํฉ๋๋ค.
ํจ์๊ฐ ๋ฆฌํด๋๋ ๊ณณ์์์ ํ์ ์ถ๋ก ์ ์ด๋ ต์ต๋๋ค. ์ด๋ค ํจ์์ ๋ฆฌํด ํ์ ์ด
impl Foo
๋ก ์ ์ธ๋์ด ์์ ๊ฒฝ์ฐ, ๊ทธ ํจ์๊ฐ ์ค์ ๋ก ๋ฆฌํดํ๋ ํ์ ์ ์์ค ์ฝ๋ ์ ์ด๋์๋ ๋ํ๋ ์์ง ์์ต๋๋ค.collect<B<() -> B
์ ๊ฐ์ด ์ ๋๋ฆญ ํ์ ์ ๋ฆฌํดํ๋ ํจ์๋B
๋ฅผ ๋ง์กฑํ๋ ์ด๋ค ํ์ ๋ ๋ฆฌํดํ ์ ์์ต๋๋ค. ์ด ๊ฒฝ์ฐ, ํธ์ถํ๋ ์ธก์์๋let x: Vec<_> = foo.collect()
๋ ํฐ๋ณดํผ์ ๋ฌธ๋ฒ์ ์จ์foo.collect::<Vec<_>>()
์ ๊ฐ์ด ๋ฆฌํด ํ์ ์ ๋ช ์์ ์ผ๋ก ์จ ์ฃผ์ด์ผ ํ ์๋ ์์ต๋๋ค.
์ด ์์๋ impl Display
๊ฐ ๋๋ฒ ์ฌ์ฉ ๋์๋ค๋ ์ ์์ ํ๋ฅญํฉ๋๋ค. ์ฌ๊ธฐ์ ์ค์ํ ๊ฒ์ ์ด ๋ impl Display
๊ฐ ์ค์ ๋ก ๊ฐ์ ํ์
์ผ ํ์๊ฐ ์๋ค๋ ๊ฒ์
๋๋ค. ๋ง์ฝ T: Display
๋ก ํธ๋ ์ ๊ฒฝ๊ณ๋ฅผ ์ ํ๊ณ ์
๋ ฅ ํ๋ผ๋ฉํฐ์ ๋ฆฌํด ๊ฐ์ ํ์
์ ๋ชจ๋ T
๋ก ํ๋ค๋ฉด, ์ด๋ ์
๋ ฅ๊ณผ ๋ฆฌํด๊ฐ์ด ๊ฐ์ ํ์
์์ ๊ฐ์ ํฉ๋๋ค. ์ด๋ ๊ฒ ํ๋ค๋ฉด ์์ ์์ ๋ ๋์ํ์ง ์์์ ๊ฒ์
๋๋ค. ์๋ํ๋ฉด, ์
๋ ฅ ๊ฐ์ ํ์
์ด format!
์ด ๋ฆฌํดํ๋ ํ์
๊ณผ ๊ฐ์ง ์์ ๊ฐ๋ฅ์ฑ์ด ๋๊ธฐ ๋๋ฌธ์
๋๋ค. ๋ง์ฝ : Display
๋ฌธ๋ฒ์ ์ฌ์ฉํ๊ณ ์ถ๋ค๋ฉด ๋
๋ฆฝ์ ์ธ ์ ๋ค๋ฆญ ๋งค๊ฐ๋ณ์๊ฐ ๋ ๊ฐ๊ฐ ํ์ํฉ๋๋ค.
์ค์ํ ํธ๋ ์
๋ฌ์คํธ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์๋ ๋ค์๊ณผ ๊ฐ์ ๋ฒ์ฉ ํธ๋ ์๋ค์ด ์ ์๋์ด ์์ต๋๋ค:
Iterator
์IntoIterator
ํธ๋ ์์for
๋ฐ๋ณต๋ฌธ์์ ์ฌ์ฉ๋ฉ๋๋ค,From
๊ณผInto
ํธ๋ ์์ ๊ฐ์ ๋ณํํ ๋ ์ฌ์ฉ๋ฉ๋๋ค,Read
์Write
ํธ๋ ์์ I/O์ ์ฌ์ฉ๋ฉ๋๋ค,Add
,Mul
๋ฑ์ ํธ๋ ์๋ค์ ์ฐ์ฐ์ ์ค๋ฒ๋ก๋ฉ(overloading)์ ์ฌ์ฉ๋ฉ๋๋ค.Drop
ํธ๋ ์์ ์๋ฉธ์ ์ ์์ ์ฌ์ฉ๋ฉ๋๋ค.Default
ํธ๋ ์์ ์ด๋ค ํ์ ์ ๊ธฐ๋ณธ๊ฐ ์ธ์คํด์ค๋ฅผ ๋ง๋ค๋ ์ฌ์ฉ๋ฉ๋๋ค.
Iterators
Iterator
ํธ๋ ์์ ์ฌ๋ฌ๋ถ์ด ์ ์ํ ํ์
์์ ๊ตฌํํด ๋ณด๊ฒ ์ต๋๋ค:
struct Fibonacci { curr: u32, next: u32, } impl Iterator for Fibonacci { type Item = u32; fn next(&mut self) -> Option<Self::Item> { let new_next = self.curr + self.next; self.curr = self.next; self.next = new_next; Some(self.curr) } } fn main() { let fib = Fibonacci { curr: 0, next: 1 }; for (i, n) in fib.enumerate().take(5) { println!("fib({i}): {n}"); } }
-
Iterator
ํธ๋ ์์ ์ปฌ๋ ์ ์ ๋ํ ์ฌ๋ฌ ์ผ๋ฐ์ ์ธ ํจ์ ํ๋ก๊ทธ๋๋ฐ ์์ (์:map
,filter
,reduce
๋ฑ)์ ๊ตฌํํฉ๋๋ค. ์ด๋ ๊ด๋ จ ๋ฌธ์๋ฅผ ๋ชจ๋ ์ฐพ์ ์ ์๋ ํธ๋ ์์ ๋๋ค. Rust์์๋ ์ด๋ฌํ ํจ์๊ฐ, ์ด์ ๋์ผํ ์ผ์ ํ๋ ๋ช ๋ นํ ๊ตฌํ๋งํผ ํจ์จ์ ์ธ ์ฝ๋๋ฅผ ์์ฑํฉ๋๋ค. -
IntoIterator
๋ ๋ฃจํ๋ฅผ ์๋ํ๊ฒ ๋ง๋๋ ํธ๋ ์์ ๋๋ค. ์ด๋Vec<T>
์ ๊ฐ์ ์ปฌ๋ ์ ์ ํ๊ณผ&Vec<T>
๋ฐ&[T]
์ ๊ฐ์ ์ด์ ๋ํ ์ฐธ์กฐ์ ์ํด ๊ตฌํ๋ฉ๋๋ค. ๋ฒ์๋ ์ด๋ฅผ ๊ตฌํํฉ๋๋ค. ์ด๋ฐ ์ด์ ๋กfor i in some_vec { .. }
๋ฅผ ์ฌ์ฉํ์ฌ ๋ฒกํฐ๋ฅผ ๋ฐ๋ณตํ ์ ์์ง๋งsome_vec.next()
๋ ์กด์ฌํ์ง ์์ต๋๋ค.
FromIterator
์ด๋ค ์ปฌ๋ ์
์ด FromIterator
๋ฅผ ๊ตฌํํ๊ณ ์๋ค๋ฉด Iterator
๋ก๋ถํฐ ๊ทธ ์ปฌ๋ ์
์ ๋ง๋ค ์ ์์ต๋๋ค.
fn main() { let primes = vec![2, 3, 5, 7]; let prime_squares = primes .into_iter() .map(|prime| prime * prime) .collect::<Vec<_>>(); }
Iterator
์๋ ๋ค์ ํจ์๊ฐ ์ ์๋์ด ์์ต๋๋ค: fn collect<B>(self) -> B where B: FromIterator<Self::Item>, Self: Sized
Iterator<Item = Result<V, E>>
์ Result<Vec<V>, E>
๋ก ๋ณํํ ์ ์๋ ๋ฉ์ง ๊ธฐ๋ฅ๋ค๋ ๊ตฌํ๋์ด ์์ต๋๋ค.
From
๊ณผ Into
ํ์
์ ์ฉ์ดํ ํ๋ณํ์ ์ํด From
๊ณผ Into
๋ฅผ ๊ตฌํํฉ๋๋ค:
fn main() { let s = String::from("hello"); let addr = std::net::Ipv4Addr::from([127, 0, 0, 1]); let one = i16::from(true); let bigger = i32::from(123i16); println!("{s}, {addr}, {one}, {bigger}"); }
From
์ด ๊ตฌํ๋๋ฉด Into
์ญ์ ์๋์ผ๋ก ๊ตฌํ๋ฉ๋๋ค:
fn main() { let s: String = "hello".into(); let addr: std::net::Ipv4Addr = [127, 0, 0, 1].into(); let one: i16 = true.into(); let bigger: i32 = 123i16.into(); println!("{s}, {addr}, {one}, {bigger}"); }
- ๊ทธ๋ ๊ธฐ ๋๋ฌธ์ ์ฌ์ฉ์ ์ ์ ํ์
์ ๊ฒฝ์ฐ์๋
From
๋ง ๊ตฌํํ๋ ๊ฒ์ด ์ผ๋ฐ์ ์ ๋๋ค. - โ
String
์ผ๋ก ๋ณํํ ์ ์๋ ๋ชจ๋ ๊ฒโ๊ณผ ๊ฐ์ ํจ์์ ์ธ์ ํ์ ์ ์ ์ธํ ๋์๋Into
๋ฅผ ์ฌ์ฉํด์ผ ํจ์ ์กฐ์ฌํ์ธ์. ๊ทธ๋์ผ๋ง, ํจ์๋From
์ ๊ตฌํํ ํ์ ๊ณผInto
๋ง ๊ตฌํํ ํ์ ๋ชจ๋๋ฅผ ์ธ์๋ก ๋ฐ์ ์ ์์ต๋๋ค.
Read
์ Write
Read
์ BufRead
๋ฅผ ์ฌ์ฉํ๋ฉด u8
ํ์
์ ๋ฐ์ดํฐ ์คํธ๋ฆผ์ ์ฝ์ ์ ์์ต๋๋ค:
use std::io::{BufRead, BufReader, Read, Result}; fn count_lines<R: Read>(reader: R) -> usize { let buf_reader = BufReader::new(reader); buf_reader.lines().count() } fn main() -> Result<()> { let slice: &[u8] = b"foo\nbar\nbaz\n"; println!("lines in slice: {}", count_lines(slice)); let file = std::fs::File::open(std::env::current_exe()?)?; println!("lines in file: {}", count_lines(file)); Ok(()) }
์ด์ ๋น์ทํ๊ฒ, Write
๋ฅผ ์ฌ์นํ๋ฉด u8
ํ์
์ ๋ฐ์ดํฐ๋ฅผ ์ธ ์ ์์ต๋๋ค:
use std::io::{Result, Write}; fn log<W: Write>(writer: &mut W, msg: &str) -> Result<()> { writer.write_all(msg.as_bytes())?; writer.write_all("\n".as_bytes()) } fn main() -> Result<()> { let mut buffer = Vec::new(); log(&mut buffer, "Hello")?; log(&mut buffer, "World")?; println!("Logged: {:?}", buffer); Ok(()) }
Drop
ํธ๋ ์
Drop
ํธ๋ ์์ ๊ตฌํํ๋ฉด, ๊ทธ ๊ฐ์ด ์ค์ฝํ ๋ฐ์ผ๋ก ๋๊ฐ ๋ ์คํ๋ ์ฝ๋๋ฅผ ์์ฑํ ์ ์์ต๋๋ค:
struct Droppable { name: &'static str, } impl Drop for Droppable { fn drop(&mut self) { println!("Dropping {}", self.name); } } fn main() { let a = Droppable { name: "a" }; { let b = Droppable { name: "b" }; { let c = Droppable { name: "c" }; let d = Droppable { name: "d" }; println!("Exiting block B"); } println!("Exiting block A"); } drop(a); println!("Exiting main"); }
drop
is called automatically, but it can be called manually like in this example.- If it was called manually, it wonโt be called at the end of the scope for the second time.
- Calling
drop
can be useful for objects that do some work ondrop
: releasing locks, closing files, etc.
๋ ผ์์ :
Drop::drop
์ ์self
๋ฅผ ์ธ์๋ก ๋ฐ์ง ์์ต๋๊น?- ์งง์ ๋๋ต: ๋ง์ฝ ๊ทธ๋ ๊ฒ ๋๋ค๋ฉด
std::mem::drop
์ด ๋ธ๋ก์ ๋์์ ํธ์ถ๋๊ณ , ๋ค์Drop::drop
์ ํธ์ถํ๊ฒ ๋์ด, ์คํ ์ค๋ฒํ๋ก๊ฐ ๋ฐ์ํฉ๋๋ค!
- ์งง์ ๋๋ต: ๋ง์ฝ ๊ทธ๋ ๊ฒ ๋๋ค๋ฉด
drop(a)
๋ฅผa.drop()
๋ก ๋ณ๊ฒฝํด ๋ณด์๊ธฐ ๋ฐ๋๋๋ค.
Default
ํธ๋ ์
Default
ํธ๋ ์์ ์ด๋ค ํ์
์ ๋ํ ๊ธฐ๋ณธ๊ฐ์ ์ ๊ณตํฉ๋๋ค.
#[derive(Debug, Default)] struct Derived { x: u32, y: String, z: Implemented, } #[derive(Debug)] struct Implemented(String); impl Default for Implemented { fn default() -> Self { Self("John Smith".into()) } } fn main() { let default_struct = Derived::default(); println!("{default_struct:#?}"); let almost_default_struct = Derived { y: "Y is set!".into(), ..Derived::default() }; println!("{almost_default_struct:#?}"); let nothing: Option<Derived> = None; println!("{:#?}", nothing.unwrap_or_default()); }
- ํธ๋ ์์ ์ง์ ๊ตฌํํ๊ฑฐ๋
#[derive(Default)]
๋ฅผ ๋ถ์ฌ์ ์ปดํ์ผ๋ฌ์๊ฒ ๊ตฌํ์ ๋งก๊ธธ ์ ์์ต๋๋ค. - ์ปดํ์ผ๋ฌ๊ฐ ์ ๊ณตํ๋ ์๋ ๊ตฌํ์ ๊ฒฝ์ฐ ๋ชจ๋ ํ๋์ ๋ํด ๊ธฐ๋ณธ ๊ฐ์ ์ค์ ํ ์ ์ธ์คํด์ค๋ฅผ ๋ฐํํฉ๋๋ค.
- ์ด๋ ๊ตฌ์กฐ์ฒด์ ๊ฐ ํ๋ ํ์
๋ค์ด ๋ชจ๋
Default
ํธ๋ ์์ ๊ตฌํํด์ผ ํจ์ ์๋ฏธํฉ๋๋ค.
- ์ด๋ ๊ตฌ์กฐ์ฒด์ ๊ฐ ํ๋ ํ์
๋ค์ด ๋ชจ๋
- ๋ฌ์คํธ ํ์ค ํ์
๋ค์ ๋๋ถ๋ถ
Default
๋ฅผ ๊ตฌํํ๊ณ ์์ผ๋ฉฐ, ๊ธฐ๋ณธ ๊ฐ์0
์ด๋""
์ฒ๋ผ ์์ ๊ฐ๋ฅํ ๊ฐ๋ค์ ๋๋ค. - ๊ตฌ์กฐ์ฒด์ ์ผ๋ถ๋ถ๋ง ๋ณต์ฌํ๊ณ ์ถ์ ๋
default
๋ฅผ ํธ๋ฆฌํ๊ฒ ์ฌ์ฉํ ์ ์์ต๋๋ค. - ๋ฌ์คํธ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋
Default
ํธ๋ ์์ ๊ตฌํํ ํ์ ์ ์ํ ํธ์ ๋ฉ์๋๋ฅผ ์ ๊ณตํ๊ธฐ๋ ํฉ๋๋ค. - ์ด
..
๋ฌธ๋ฒ์ ๊ตฌ์กฐ์ฒด ์ ๋ฐ์ดํธ ๋ฌธ๋ฒ(struct update syntax)๋ผ๊ณ ํฉ๋๋ค
Add
, Mul
, โฆ
์ฐ์ฐ์ ์ค๋ฒ๋ก๋๋ std::ops
์ ์๋ ๋ค์ํ ํธ๋ ์๋ค์ ํตํด ๊ตฌํ๋ฉ๋๋ค:
#[derive(Debug, Copy, Clone)] struct Point { x: i32, y: i32 } impl std::ops::Add for Point { type Output = Self; fn add(self, other: Self) -> Self { Self {x: self.x + other.x, y: self.y + other.y} } } fn main() { let p1 = Point { x: 10, y: 20 }; let p2 = Point { x: 100, y: 200 }; println!("{:?} + {:?} = {:?}", p1, p2, p1 + p2); }
๋ ผ์์ :
&Point
๊ฐAdd
๋ฅผ ๊ตฌํํ๋๋ก ํ ์๋ ์์ต๋๋ค. ์ด๊ฒ ์ด๋ค ๊ฒฝ์ฐ์ ์ ์ฉํ ๊น์?- ๋ต:
Add:add
๋self
๋ฅผ ์๋ชจํฉ๋๋ค. ๋ง์ฝ ํ์T
๊ฐCopy
ํธ๋ ์์ ๊ตฌํํ๊ณ ์์ง ์๋ค๋ฉด&T
์ ๋ํด์๋ ์ฐ์ฐ์ ์ค๋ฒ๋ก๋ฉ์ ๊ณ ๋ คํด์ผ ํฉ๋๋ค. ์ด๋ ๊ฒ ํ๋ฉด ํธ์ถ๋ถ์์ ๋ถํ์ํ ๋ณต์ฌ๋ฅผ ํผํ ์ ์์ต๋๋ค.
- ๋ต:
- ์
Output
์ด ์ฐ๊ด๋ ํ์ ์ธ๊ฐ์? ํ์ ํ๋ผ๋ฉํฐ๋ก ๋ง๋ค ์ ์์๊น์?- ๋ต: ํ์
ํ๋ผ๋ฉํฐ๋ฅผ ํธ์ถํ๋ ์ชฝ์์ ๊ฒฐ์ ํฉ๋๋ค. ๋ฐ๋ฉด ์ฐ๊ด๋ ํ์
(
Output
๊ฐ์) ์ ํธ๋ ์์ ๊ตฌํํ๋ ์ชฝ์์ ์ ์ด ๊ฐ๋ฅํฉ๋๋ค.
- ๋ต: ํ์
ํ๋ผ๋ฉํฐ๋ฅผ ํธ์ถํ๋ ์ชฝ์์ ๊ฒฐ์ ํฉ๋๋ค. ๋ฐ๋ฉด ์ฐ๊ด๋ ํ์
(
Add
๋ฅผ ์ด์ฉํด์ ์๋ก ๋ค๋ฅธ ๋ ๊ฐ์ ํ์ ์ ๋ํ ์๋ ์์ต๋๋ค. ์๋ฅผ ๋ค์ดimpl Add<(i32, i32)> for Point
๋ ํํ์Point
์ ๋ํ ์ ์๊ฒ ํด ์ค๋๋ค.
ํด๋ก์ (Closure)
ํด๋ก์ ํน์ ๋๋คํํ์์ ์ต๋ช
ํ์
์
๋๋ค. ์ด๋ค์ Fn
,FnMut
, FnOnce
๋ผ๋ ํน๋ณํ ํธ๋ ์์ ๊ตฌํํฉ๋๋ค:
fn apply_with_log(func: impl FnOnce(i32) -> i32, input: i32) -> i32 { println!("Calling function on {input}"); func(input) } fn main() { let add_3 = |x| x + 3; println!("add_3: {}", apply_with_log(add_3, 10)); println!("add_3: {}", apply_with_log(add_3, 20)); let mut v = Vec::new(); let mut accumulate = |x: i32| { v.push(x); v.iter().sum::<i32>() }; println!("accumulate: {}", apply_with_log(&mut accumulate, 4)); println!("accumulate: {}", apply_with_log(&mut accumulate, 5)); let multiply_sum = |x| x * v.into_iter().sum::<i32>(); println!("multiply_sum: {}", apply_with_log(multiply_sum, 3)); }
Fn
(์๋ฅผ ๋ค์ด add_3
)์ ์บก์ฒ๋ ๊ฐ์ ์๋ชจ๋ ๋ณ๊ฒฝ๋ ํ์ง ์๊ณ , ํน์ ์ด๋ค ๊ฒ๋ ์บก์ณํ์ง ์์์ ์๋ ์๊ธฐ ๋๋ฌธ์ ๋์์ ์ฌ๋ฌ๋ฒ ํธ์ถํ ์ ์์ต๋๋ค.
FnMut
(์๋ฅผ ๋ค์ด accumulate
)๋ ์บก์ฒ๋ ๊ฐ์ ๋ณ๊ฒฝํ ์ ์์ผ๋ฏ๋ก ์ฌ๋ฌ ๋ฒ ํธ์ถ์ ๊ฐ๋ฅํ์ง๋ง ๋์์ ํธ์ถ ํ ์๋ ์์ต๋๋ค.
FnOnce
(์๋ฅผ ๋ค์ด multiply_sum
)๋ ํ๋ฒ๋ง ํธ์ถ๋๋ฉฐ ์บก์ฒ๋ ๊ฐ์ ์๋ชจํฉ๋๋ค.
FnMut
๋ FnOnce
์ ํ์ํ์
์
๋๋ค. Fn
์ FnMut
๊ณผ FnOnce
์ ํ์ ํ์
์
๋๋ค. ์ฆ, FnMut
๋ FnOnce
๊ฐ ํธ์ถ๋๋ ๊ณณ์ด๋ฉด ์ด๋์๋ ์ฌ์ฉ ํ ์ ์๊ณ Fn
์ FnMut
์ FnOnce
๊ฐ ํธ์ถ๋๋ ๊ณณ์ด๋ฉด ์ด๋๋ ์ฌ์ฉํ ์ ์์ต๋๋ค.
์ปดํ์ผ๋ฌ๋ ํด๋ก์ ๊ฐ ๋ฌด์์ ์บก์ณํ๋์ง์ ๋ฐ๋ผ Copy
(์๋ฅผ ๋ค์ด add_3
)๊ณผ Clone
(์๋ฅผ ๋ค์ด multiply_sum
)์ ์์์ ์ถ๋ก ํฉ๋๋ค.
๊ธฐ๋ณธ์ ์ผ๋ก ํด๋ก์ ธ๋, ๊ฐ๋ฅํ๋ค๋ฉด, ์ฐธ์กฐ๋ฅผ ์ฌ์ฉํ์ฌ ์บก์ณ๋ฅผ ํฉ๋๋ค. move
ํค์๋๋ฅผ ์ฐ๋ฉด ๊ฐ์ผ๋ก ์บก์ณ๊ฐ ๋ฉ๋๋ค.
fn make_greeter(prefix: String) -> impl Fn(&str) { return move |name| println!("{} {}", prefix, name) } fn main() { let hi = make_greeter("Hi".to_string()); hi("there"); }
3์ผ์ฐจ ์ค์ ์ฐ์ต๋ฌธ์
We will design a classical GUI library using traits and trait objects.
We will also look at enum dispatch with an exercise involving points and polygons.
After looking at the exercises, you can look at the solutions provided.
๊ฐ๋จํ GUI ๋ผ์ด๋ธ๋ฌ๋ฆฌ
์ด๋ฒ ์ฐ์ต๋ฌธ์ ์์๋ ํธ๋ ์์ ํธ๋ ์ ๊ฐ์ฒด์ ๋ํด ๋ฐฐ์ด๊ฒ์ ํ์ฉํ์ฌ ๊ณ ์ ์ ์ธ GUI ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ค๊ณํ ๊ฒ์ ๋๋ค.
๋ผ์ด๋ธ๋ฌ๋ฆฌ์๋ ๋ช ๊ฐ์ง ์์ ฏ์ด ์์ต๋๋ค:
Window
:title
์์ฑ์ ๊ฐ์ง๊ณ ์์ผ๋ฉฐ, ๋ค๋ฅธ ์์ ฏ๋ค์ ํฌํจํ ์ ์์ต๋๋ค.Button
:label
์์ฑ์ ๊ฐ์ง๊ณ ์์ผ๋ฉฐ, ๋ฒํผ์ด ๋๋ ธ์๋ ์คํ๋๋ ์ฝ๋ฐฑ ํจ์๊ฐ ์์ต๋๋ค.Label
:label
์์ฑ์ ๊ฐ์ง๊ณ ์์ต๋๋ค.
์์ ฏ๋ค์ ๋ชจ๋ Widget
ํธ๋ ์์ ๊ตฌํํฉ๋๋ค. ์๋ ์ฝ๋๋ฅผ ์ฐธ์กฐํ์ธ์.
์๋ ์ฝ๋๋ฅผ https://play.rust-lang.org/์ ๋ณต์ฌํ๊ณ ๋๋ฝ๋ draw_into
๋ฉ์๋๋ฅผ ์ฑ์ ๋ฃ์ด Widget
ํธ๋ ์์ ์์ฑํด๋ด
์๋ค:
// TODO: remove this when you're done with your implementation. #![allow(unused_imports, unused_variables, dead_code)] pub trait Widget { /// Natural width of `self`. fn width(&self) -> usize; /// Draw the widget into a buffer. fn draw_into(&self, buffer: &mut dyn std::fmt::Write); /// Draw the widget on standard output. fn draw(&self) { let mut buffer = String::new(); self.draw_into(&mut buffer); println!("{buffer}"); } } pub struct Label { label: String, } impl Label { fn new(label: &str) -> Label { Label { label: label.to_owned(), } } } pub struct Button { label: Label, callback: Box<dyn FnMut()>, } impl Button { fn new(label: &str, callback: Box<dyn FnMut()>) -> Button { Button { label: Label::new(label), callback, } } } pub struct Window { title: String, widgets: Vec<Box<dyn Widget>>, } impl Window { fn new(title: &str) -> Window { Window { title: title.to_owned(), widgets: Vec::new(), } } fn add_widget(&mut self, widget: Box<dyn Widget>) { self.widgets.push(widget); } fn inner_width(&self) -> usize { std::cmp::max( self.title.chars().count(), self.widgets.iter().map(|w| w.width()).max().unwrap_or(0), ) } } impl Widget for Label { fn width(&self) -> usize { unimplemented!() } fn draw_into(&self, buffer: &mut dyn std::fmt::Write) { unimplemented!() } } impl Widget for Button { fn width(&self) -> usize { unimplemented!() } fn draw_into(&self, buffer: &mut dyn std::fmt::Write) { unimplemented!() } } impl Widget for Window { fn width(&self) -> usize { unimplemented!() } fn draw_into(&self, buffer: &mut dyn std::fmt::Write) { unimplemented!() } } fn main() { let mut window = Window::new("Rust GUI Demo 1.23"); window.add_widget(Box::new(Label::new("This is a small text GUI demo."))); window.add_widget(Box::new(Button::new( "Click me!", Box::new(|| println!("You clicked the button!")), ))); window.draw(); }
์ ํ๋ก๊ทธ๋จ์ ์ถ๋ ฅ์ ์๋์ ๊ฐ์ต๋๋ค:
========
Rust GUI Demo 1.23
========
This is a small text GUI demo.
| Click me! |
ํ
์คํธ๋ฅผ ์ค๋ง์ถค ํด์ ๊ทธ๋ฆฌ๋ ค๋ฉด fill/alignment๋ฅผ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค. ํน์ ๋ฌธ์(์ฌ๊ธฐ์๋ '/'
)๋ก ํจ๋ฉ์ ์ฃผ๋ ๋ฐฉ๋ฒ๊ณผ ์ ๋ ฌ์ ์ ์ดํ๋ ๋ฐฉ๋ฒ์ ํ์ธํ์๊ธฐ ๋ฐ๋๋๋ค:
fn main() { let width = 10; println!("left aligned: |{:/<width$}|", "foo"); println!("centered: |{:/^width$}|", "foo"); println!("right aligned: |{:/>width$}|", "foo"); }
์์ ์ ๋ ฌ ํธ๋ฆญ์ ์ฌ์ฉํ์ฌ ๋ค์๊ณผ ๊ฐ์ ์ถ๋ ฅ์ ์์ฑํ ์ ์์ต๋๋ค:
+--------------------------------+
| Rust GUI Demo 1.23 |
+================================+
| This is a small text GUI demo. |
| +-----------+ |
| | Click me! | |
| +-----------+ |
+--------------------------------+
Polygon ๊ตฌ์กฐ์ฒด
์ฐ๋ฆฌ๋ ๋ช๊ฐ์ ๊ผญ์ง์ ์ ๊ฐ์ง ๋ค๊ฐํ์ ํํํ๋ Polygon
๊ตฌ์กฐ์ฒด๋ฅผ ๋ง๋ค ๊ฒ์
๋๋ค. ์๋ ์ฝ๋๋ฅผ https://play.rust-lang.org/์ ๋ณต์ฌํด์ ํ
์คํธ๊ฐ ํต๊ณผํ๋๋ก ๋น ์ง ๋ฉ์๋๋ฅผ ๊ตฌํํ์๋ฉด ๋ฉ๋๋ค:
// TODO: remove this when you're done with your implementation. #![allow(unused_variables, dead_code)] pub struct Point { // add fields } impl Point { // add methods } pub struct Polygon { // add fields } impl Polygon { // add methods } pub struct Circle { // add fields } impl Circle { // add methods } pub enum Shape { Polygon(Polygon), Circle(Circle), } #[cfg(test)] mod tests { use super::*; fn round_two_digits(x: f64) -> f64 { (x * 100.0).round() / 100.0 } #[test] fn test_point_magnitude() { let p1 = Point::new(12, 13); assert_eq!(round_two_digits(p1.magnitude()), 17.69); } #[test] fn test_point_dist() { let p1 = Point::new(10, 10); let p2 = Point::new(14, 13); assert_eq!(round_two_digits(p1.dist(p2)), 5.00); } #[test] fn test_point_add() { let p1 = Point::new(16, 16); let p2 = p1 + Point::new(-4, 3); assert_eq!(p2, Point::new(12, 19)); } #[test] fn test_polygon_left_most_point() { let p1 = Point::new(12, 13); let p2 = Point::new(16, 16); let mut poly = Polygon::new(); poly.add_point(p1); poly.add_point(p2); assert_eq!(poly.left_most_point(), Some(p1)); } #[test] fn test_polygon_iter() { let p1 = Point::new(12, 13); let p2 = Point::new(16, 16); let mut poly = Polygon::new(); poly.add_point(p1); poly.add_point(p2); let points = poly.iter().cloned().collect::<Vec<_>>(); assert_eq!(points, vec![Point::new(12, 13), Point::new(16, 16)]); } #[test] fn test_shape_perimeters() { let mut poly = Polygon::new(); poly.add_point(Point::new(12, 13)); poly.add_point(Point::new(17, 11)); poly.add_point(Point::new(16, 16)); let shapes = vec![ Shape::from(poly), Shape::from(Circle::new(Point::new(10, 20), 5)), ]; let perimeters = shapes .iter() .map(Shape::perimeter) .map(round_two_digits) .collect::<Vec<_>>(); assert_eq!(perimeters, vec![15.48, 31.42]); } } #[allow(dead_code)] fn main() {}
๋๋ฝ๋ ๋ฉ์๋ ์๊ทธ๋์ฒ๋ฅผ ์ฌ๋ฐ๋ฅด๊ฒ ์ ์ํ๋ ๊ฒ์ด ๋ฌธ์ ์ ํต์ฌ ๋ถ๋ถ์ ๋๋ค. ํ ์คํธ๋ ์์ ํ๋ฉด ์๋ฉ๋๋ค.
์ฐ์ต๋ฌธ์ ์ ๋ค๋ฅธ ํฅ๋ฏธ๋ก์ด ๋ถ๋ถ:
- ํ
์คํธ ์ฝ๋๋ฅผ ๋ณด๋ฉด ์ด๋ค ๋ฉ์๋๋ค์ ์ธ์๋ฅผ borrowํ๋ ๋์
Copy
ํธ๋ ์์ ์ฌ์ฉํ๊ธฐ๋ ํฉ๋๋ค. ๊ตฌ์กฐ์ฒด๊ฐCopy
ํธ๋ ์์ ์์(derive)ํ๋๋ก ํ๋ฉด ๋ฉ๋๋ค. - โ+โ๋ฅผ ์ฌ์ฉํ์ฌ ๋ ๊ฐ์ฒด๋ฅผ ์๋ก ๋ํ๋ ค๋ฉด
Add
ํธ๋ ์์ ๊ตฌํํด์ผ ํฉ๋๋ค. ์ด๋ 3์ผ์ฐจ์ ๋ค๋ฃฐ ๋ด์ฉ์ ๋๋ค.
์ค๋ฅ์ฒ๋ฆฌ
๋ฌ์คํธ์์ ์ค๋ฅ๋ ๋ช ์์ ์ธ ํ๋ฆ์ ๋ฐ๋ผ ์ฒ๋ฆฌ๊ฐ ๋ฉ๋๋ค:
- ์ค๋ฅ๋ฅผ ๋ฐ์ํ ์ ์๋ ํจ์๋ ๋ฐํ ํ์ ์ ์ด๋ฅผ ๋ช ์ํด์ผ ํฉ๋๋ค,
- ์์ธ(exception) ๊ธฐ๋ฅ์ ์์ต๋๋ค.
ํจ๋
๋ฌ์คํธ๋ ์ํ ์ค ์น๋ช ์ ์ธ ์ค๋ฅ๋ฅผ ๋ง๋๋ฉด ํจ๋์ ๋ฐ์ํ ๊ฒ์ ๋๋ค:
fn main() { let v = vec![10, 20, 30]; println!("v[100]: {}", v[100]); }
- ํจ๋์ ๋ณต๊ตฌํ ์ ์๊ณ ์์์น ๋ชปํ ์ค๋ฅ์
๋๋ค.
- ํจ๋์ ํ๋ก๊ทธ๋จ์ ๋ฒ๊ทธ๊ฐ ์๋ค๋ ๊ฒ์ ๋ํ๋ ๋๋ค.
- ์ถฉ๋(ํฌ๋์)์ ํ์ฉํ์ง ์์์ผ ํ๋ ๊ฒฝ์ฐ, ํจ๋์ ์ ๋ฐํ์ง ์๋ API(
Vec::get
๋ฑ)๋ฅผ ์ฌ์ฉํ๋ฉด ๋ฉ๋๋ค.
์คํ ๋๊ฐ๊ธฐ
๊ธฐ๋ณธ์ ์ผ๋ก, ํจ๋์ด ๋ฐ์ํ๋ฉด ์คํ ๋๊ฐ๊ธฐ๋ฅผ ํฉ๋๋ค. ์คํ ๋๊ฐ๊ธฐ๋ ๋ค์๊ณผ ๊ฐ์ด ์บ์น๊ฐ ๊ฐ๋ฅํฉ๋๋ค:
use std::panic; fn main() { let result = panic::catch_unwind(|| { println!("hello!"); }); assert!(result.is_ok()); let result = panic::catch_unwind(|| { panic!("oh no!"); }); assert!(result.is_err()); }
- ์ด๊ฒ์ ๋จ์ผ ์์ฒญ์ด ํฌ๋์ ๋๋๋ผ๋ ํ๋ก๊ทธ๋จ์ด ๊ณ์ ์คํ๋์ผ ํ๋ ์๋ฒ์ ์ ์ฉํฉ๋๋ค.
- ๋ง์ฝ
Cargo.toml
์ค์ ํ์ผ์panic = abort
์ ์ค์ ํ๋ค๋ฉด ํฌ๋์๋ฅผ ์บ์นํ ์ ์์ต๋๋ค.
Result
๋ฅผ ์ด์ฉํ ๊ตฌ์กฐํ๋ ์ค๋ฅ์ฒ๋ฆฌ
์ฌ๋ฌ๋ถ์ ์ด๋ฏธ Result
์ด๊ฑฐํ์ ๋ช ๋ฒ ๋ดค์ต๋๋ค. ์ด ํ์
์ ํ๋ก๊ทธ๋จ์ ์ ์์ ์ธ ์ํ ์ค์ ๋ฐ์ํ ์ ์๋ ์ค๋ฅ๊ฐ๋ค์ ๋ํ๋ด๊ธฐ ์ํด ์ฌ์ฉ๋ฉ๋๋ค:
use std::fs; use std::io::Read; fn main() { let file = fs::File::open("diary.txt"); match file { Ok(mut file) => { let mut contents = String::new(); file.read_to_string(&mut contents); println!("Dear diary: {contents}"); }, Err(err) => { println!("The diary could not be opened: {err}"); } } }
Option
์ ๋ง์ฐฌ๊ฐ์ง๋ก, ์ฑ๊ณตํ ๊ฒฝ์ฐ์ ๊ฐ์Result
๋ด๋ถ์ ์์ต๋๋ค. ๊ทธ๋์, ๊ฐ๋ฐ์๋ ๋ช ์์ ์ผ๋ก ์ด๋ฅผ ์ถ์ถํ์ฌ์ผ ํฉ๋๋ค. ์ด๋ ๊ฒ ํจ์ผ๋ก์จ ๊ฐ์ ์ฝ๊ธฐ ์ ์ ์ค๋ฅ ๋ฐ์ ์ฌ๋ถ๋ฅผ ๋ฐ๋์ ์ฒดํฌํ๋๋ก ์ ๋ํ๊ณ ์์ต๋๋ค. ๋ง์ผ ์ค๋ฅ๊ฐ ์ ๋ ๋ฐ์ํ์ง ์๋ ๊ฒฝ์ฐ๋ผ๋ฉดunwrap()
์ด๋expect()
๋ฅผ ์ฌ์ฉํ ์ ์์ผ๋ฉฐ, ์ด๋ ๊ฐ๋ฐ์์ ์๋(์ญ์ฃผ: ์ค๋ฅ๊ฐ ๋ฐ์ํ ์ ์์)์ ๋ช ์์ ์ผ๋ก ๋ํ๋ด๋ ๋ฐฉ๋ฒ์ด๊ธฐ๋ ํฉ๋๋ค.- ์์
์ค์ ์๋์ง๋ง
Result
์ API ๋ ํผ๋ฐ์ค๋ฅผ ์ฝ๋ ๊ฒ์ ๊ถ์ฅํฉ๋๋ค. ํจ์ํ ํ๋ก๊ทธ๋๋ฐ ์คํ์ผ์ ๋์์ด ๋๋ ํธ๋ฆฌํ ๋ฉ์๋์ ํจ์๋ฅผ ๋ง์ด ๋ฐฐ์ธ ์ ์์ต๋๋ค.
?
๋ฅผ ์ด์ฉํ ์ค๋ฅ ์ ํ
์ฐ์ฐ์ ?
๋ ํธ์ถ์์๊ฒ ์ค๋ฅ๋ฅผ ๋ฐํํ ๋ ์ฌ์ฉํฉ๋๋ค. ์ด๋ฅผ ์ด์ฉํ๋ฉด ์ด๋ฐ ์ฝ๋๋ฅผ
match some_expression {
Ok(value) => value,
Err(err) => return Err(err),
}
์ด๋ ๊ฒ ์งง๊ฒ ์ธ ์ ์์ต๋๋ค
some_expression?
์ด์ ์ฐ๋ฆฌ ์์ ์ ์ ์ฉํด ๋ณด๊ฒ ์ต๋๋ค:
use std::{fs, io}; use std::io::Read; fn read_username(path: &str) -> Result<String, io::Error> { let username_file_result = fs::File::open(path); let mut username_file = match username_file_result { Ok(file) => file, Err(err) => return Err(err), }; let mut username = String::new(); match username_file.read_to_string(&mut username) { Ok(_) => Ok(username), Err(err) => Err(err), } } fn main() { //fs::write("config.dat", "alice").unwrap(); let username = read_username("config.dat"); println!("username or error: {username:?}"); }
ํค ํฌ์ธํธ:
username
๋ณ์๋Ok(string)
์ด๊ฑฐ๋Err(error)
์ผ ์ ์์ต๋๋ค.fs::write
๋ฉ์๋๋ฅผ ์ฌ์ฉํ์ฌ ํ์ผ์ด ์๊ฑฐ๋, ๋น์๊ฑฐ๋, ์ค๋ณต๋๋ ๊ฒฝ์ฐ ๋ฑ์ ํ ์คํธํด ๋ด ๋๋ค.- ํจ์์ ๋ฆฌํด ํ์
์ ๋ค์คํ
๋์ด ํธ์ถ๋๋ ํจ์์ ๋ฆฌํด ํ์
๊ณผ ํธํ๋์ด์ผ ํฉ๋๋ค. ์๋ฅผ ๋ค์ด
Result<T, Err>
๋ฅผ ๋ฆฌํดํ๋ ํจ์๋Result<AnyT, Err>
๋ฅผ ๋ฆฌํดํ๋ ํจ์๋ฅผ ํธ์ถํ ๋์๋ง?
๋ฅผ ์ฌ์ฉํ ์ ์์ต๋๋ค.Option<AnyT>
๋Result<T, OtherErr>
(OtherError
๊ฐFrom<Err>
๋ฅผ ๊ตฌํํ์ง ์๋ ๋ค๊ณ ๊ฐ์ ํ ๋)์ ๊ฐ์ ํ์ ์ ๋ฆฌํดํ๋ ํจ์๋ฅผ ํธ์ถํ ๋์๋?
๋ฅผ ์ฌ์ฉํ ์ ์์ต๋๋ค. ๋์ผํ ์ด์ ๋ก,Option<T>
๋ฅผ ๋ฆฌํดํ๋ ํจ์๋Option<AnyT>
๋ฅผ ๋ฆฌํดํ๋ ํจ์๋ฅผ ํธ์ถํ ๋์๋ง?
๋ฅผ ์ฌ์ฉํ ์ ์์ต๋๋ค.Option
๊ณผResult
๊ฐ์ ๋ณํ์ ์ํดOption::ok_or
,Result::ok
,Result::err
์ ๊ฐ์ ํจ์๋ค์ ์ฌ์ฉํ ์ ์์ต๋๋ค.
์ค๋ฅํ์ ๋ณํ
์ค์ ๋ก ?
๊ฐ ์ ์ฉ๋๋ ๊ณผ์ ์ ์๊น ์ค๋ช
ํ ๊ฒ ๋ณด๋ค ์ข ๋ ๋ณต์กํฉ๋๋ค:
expression?
์ ํํ์ ์๋์ ๊ฐ์ต๋๋ค
match expression {
Ok(value) => value,
Err(err) => return Err(From::from(err)),
}
From::from
์ ํตํด ์๋์ ์๋ฌ ํ์
์ ์ด ํจ์๊ฐ ๋ฐํํ๋ ์๋ฌ ํ์
์ผ๋ก ๋ณํํ๊ณ ์์ต๋๋ค:
์ค๋ฅํ์ ๋ณํ
use std::error::Error; use std::fmt::{self, Display, Formatter}; use std::fs::{self, File}; use std::io::{self, Read}; #[derive(Debug)] enum ReadUsernameError { IoError(io::Error), EmptyUsername(String), } impl Error for ReadUsernameError {} impl Display for ReadUsernameError { fn fmt(&self, f: &mut Formatter) -> fmt::Result { match self { Self::IoError(e) => write!(f, "IO error: {e}"), Self::EmptyUsername(filename) => write!(f, "Found no username in {filename}"), } } } impl From<io::Error> for ReadUsernameError { fn from(err: io::Error) -> ReadUsernameError { ReadUsernameError::IoError(err) } } fn read_username(path: &str) -> Result<String, ReadUsernameError> { let mut username = String::with_capacity(100); File::open(path)?.read_to_string(&mut username)?; if username.is_empty() { return Err(ReadUsernameError::EmptyUsername(String::from(path))); } Ok(username) } fn main() { //fs::write("config.dat", "").unwrap(); let username = read_username("config.dat"); println!("username or error: {username:?}"); }
ํค ํฌ์ธํธ:
username
๋ณ์๋Ok(string)
์ด๊ฑฐ๋Err(error)
์ผ ์ ์์ต๋๋ค.fs::write
๋ฉ์๋๋ฅผ ์ฌ์ฉํ์ฌ ํ์ผ์ด ์๊ฑฐ๋, ๋น์๊ฑฐ๋, ์ค๋ณต๋๋ ๊ฒฝ์ฐ ๋ฑ์ ํ ์คํธํด ๋ด ๋๋ค.
๋ชจ๋ ์๋ฌ ํ์
(no_std
์ด์ด์ผ ํ๋ ์๋ฌ ํ์
์ ์ ์ธํ๊ณ )์ ๋ํด std::error::Error
๋ฅผ ๊ตฌํํ๋ ๊ฒ์ ์ข์ ์ต๊ด์
๋๋ค. std::error::Error
๋ฅผ ๊ตฌํํ๋ค๋ ๊ฒ์ Debug
์ Display
๋ ๊ตฌํํ๋ค๋ ๊ฒ์
๋๋ค. core
๋ฅผ ์ํ Error
ํฌ๋ ์ดํฌ๋ ๋์ดํ๋ฆฌ์๋ง ์ ๊ณต์ด ๋ฉ๋๋ค. ๊ทธ๋์ ์์ง no_std
ํ๊ฒฝ์์ ์ฌ์ฉํ ์๋ ์์ต๋๋ค.
๊ฐ๋ฅํ๋ค๋ฉด Clone
๊ณผ Eq
ํธ๋ ์๋ ๊ตฌํํ๋๋ก ํ์ธ์. ์ฌ๋ฌ๋ถ์ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๊ฐ ํ
์คํธ ํ๊ธฐ ์ฌ์์ง๊ณ , ์ฌ์ฉํ๊ธฐ ์ข์์ง ๊ฒ๋๋ค. ๋ค๋ง, ์ด ์์ ์์๋ ๊ทธ๋ ๊ฒ ํ๊ธฐ ํ๋ญ๋๋ค. ์๋ํ๋ฉด io::Error
๋ ์ด ํธ๋ ์๋ค์ ๊ตฌํํ๊ณ ์์ง ์๊ธฐ ๋๋ฌธ์
๋๋ค.
๋๋ค๋ฅธ ์ค๋ฅ ์ด๊ฑฐํ
thiserror๋, ์ด์ ํ์ด์ง์์ ๋ณด์๋ ๊ฒ๊ณผ ๊ฐ์ ์๋ฌ ์ด๊ฑฐํ์ ์ฝ๊ฒ ๋ง๋ค ์ ์๊ฒ ํด ์ฃผ๋ ์ ๋ช ํ ํฌ๋ ์ดํธ ์ ๋๋ค:
use std::{fs, io}; use std::io::Read; use thiserror::Error; #[derive(Debug, Error)] enum ReadUsernameError { #[error("Could not read: {0}")] IoError(#[from] io::Error), #[error("Found no username in {0}")] EmptyUsername(String), } fn read_username(path: &str) -> Result<String, ReadUsernameError> { let mut username = String::new(); fs::File::open(path)?.read_to_string(&mut username)?; if username.is_empty() { return Err(ReadUsernameError::EmptyUsername(String::from(path))); } Ok(username) } fn main() { //fs::write("config.dat", "").unwrap(); match read_username("config.dat") { Ok(username) => println!("Username: {username}"), Err(err) => println!("Error: {err}"), } }
thiserror
์ derive ๋งคํฌ๋ก๋ฅผ ์ด์ฉํ๋ฉด std::error::Error
๊ณผ Display
(๋ง์ฝ #[error(...)]
์ดํธ๋ฆฌ๋ทฐํธ๋ฅผ ์ถ๊ฐํ์ ๊ฒฝ์ฐ), From
(๋ง์ฝ #[from]
์ดํธ๋ฆฌ๋ทฐํธ๋ฅผ ์ถ๊ฐํ์ ๊ฒฝ์ฐ) ํธ๋ ์๋ค์ด ์๋์ผ๋ก ๊ตฌํ์ด ๋ฉ๋๋ค. ๊ตฌ์กฐ์ฒด์ ๋ํด์๋ ์ฌ์ฉ ๊ฐ๋ฅํฉ๋๋ค.
์ด ๋งคํฌ๋ก๋ฅผ ์ฌ์ฉํด๋ ๋ฐ์ผ๋ก ๋ ธ์ถ๋๋ API๊ฐ ๋ณ๊ฒฝ๋์ง๋ ์์ต๋๋ค. ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ๋ง๋ค ๊ฒฝ์ฐ์๋ ์ด๊ฒ ์ค์ํ์ฃ .
๋์ ์ธ ์๋ฌ ํ์
๋๋๋ก ์ฐ๋ฆฌ๋, ๋ฐ์ ๊ฐ๋ฅํ ๋ชจ๋ ์๋ฌ๋ฅผ ์ผ์ผํ ์ด๊ฑฐํ์ง ์๊ณ , ์ด๋ค ์ข
๋ฅ์ ์๋ฌ๋ผ๋ ์๊ด์์ด ๋ฆฌํดํ๊ณ ์ถ์ ๋๊ฐ ์์ต๋๋ค. std::error::Error
๋ฅผ ์ด์ฉํ๋ฉด ์ฝ์ต๋๋ค.
use std::fs; use std::io::Read; use thiserror::Error; use std::error::Error; #[derive(Clone, Debug, Eq, Error, PartialEq)] #[error("Found no username in {0}")] struct EmptyUsernameError(String); fn read_username(path: &str) -> Result<String, Box<dyn Error>> { let mut username = String::new(); fs::File::open(path)?.read_to_string(&mut username)?; if username.is_empty() { return Err(EmptyUsernameError(String::from(path)).into()); } Ok(username) } fn main() { //fs::write("config.dat", "").unwrap(); match read_username("config.dat") { Ok(username) => println!("Username: {username}"), Err(err) => println!("Error: {err}"), } }
์ด๋ ๊ฒ ํ๋ฉด ์ฝ๋์ ์์ ์ค์ผ ์ ์์ต๋๋ค. ๊ทธ๋ฌ๋ ์๋ก ๋ค๋ฅธ ์ข
๋ฅ์ ์๋ฌ๋ฅผ ๊ตฌ๋ณํ์ฌ ๋ค๋ฅด๊ฒ ์ฒ๋ฆฌํ๋ ๊ฒ์ด ๋ถ๊ฐ๋ฅํด ์ง๋๋ค. ๋๋ฌธ์, Box<dyn Error>
๋ฅผ ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ API๋ก ๋
ธ์ถํ๋๊ฒ ์ข์ ๋์์ธ์ ์๋๋๋ค. ์๋ฌ ๋ฐ์ ์, ๊ทธ์ ์๋ฌ ๋ฉ์์ง๋ฅผ ์ถ๋ ฅํ๊ณ ์ถ์ ๊ฒฝ์ฐ์ ๊ฐ์ด ์ ํ๋ ์ํฉ์์๋ ์ ์ฉํ ์ ์์ต๋๋ค.
์ค๋ฅ์ ์ํฉ์ ๋ณด ์ถ๊ฐ
anyhow ํฌ๋ ์ดํธ๋ ์๋ฌ์ ์๋ฌ๊ฐ ๋ฐ์ํ ๋ฌธ๋งฅ์ ๋ํ ์ ๋ณด๋ฅผ ์ถ๊ฐํ๊ธฐ ์ํด ๋๋ฆฌ ์ฌ์ฉ๋๋ฉฐ, ์ด๋ฅผ ์ด์ฉํ๋ฉด ์๋ก ๋ค๋ฅธ ๋ฌธ๋งฅ์ ๋ํ๋ด๊ธฐ ์ฌ์ฉ์ ์ ์ ์ค๋ฅ ํ์ ์ ๋ง์ด ๋ง๋ค์ด์ผ ํ๋ ๋ถํธํจ์ ํผํ ์ ์์ต๋๋ค:
use std::{fs, io}; use std::io::Read; use anyhow::{Context, Result, bail}; fn read_username(path: &str) -> Result<String> { let mut username = String::with_capacity(100); fs::File::open(path) .with_context(|| format!("Failed to open {path}"))? .read_to_string(&mut username) .context("Failed to read")?; if username.is_empty() { bail!("Found no username in {path}"); } Ok(username) } fn main() { //fs::write("config.dat", "").unwrap(); match read_username("config.dat") { Ok(username) => println!("Username: {username}"), Err(err) => println!("Error: {err:?}"), } }
anyhow::Result<V>
๋Result<V, anyhow::Error>
์ ํ์ ์จ๋ฆฌ์ด์ค(alias)์ ๋๋ค.anyhow::Error
๋Box<dyn Error>
์ ๋ํผ ํ์ ์ด๋ผ ํ ์ ์์ต๋๋ค. ๋ฐ๋ผ์ ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ๊ณต๊ฐ API๋ก์ ์ฌ์ฉํ๊ธฐ์ ๋ถ์ ํฉํ๋ค๊ณ ํ ์ ์์ง๋ง ๋ง์ ์ ํ๋ฆฌ์ผ์ด์ ์ ๋๋ฆฌ ์ฌ์ฉ๋๊ณ ์์ต๋๋ค.- ํ์ํ๋ค๋ฉด
anyhow::Error
์ ์ ์ฅ๋ ์ง์ง ์๋ฌ ํ์ ์ ๊บผ๋ด์ด ๊ฒ์ฌํ ์๋ ์์ต๋๋ค. anyhow::Result<T>
๊ฐ ์ ๊ณตํ๋ ๊ธฐ๋ฅ๋ค์ด Go ์ธ์ด ๊ฐ๋ฐ์๋ค์๊ฒ๋ ์ต์ํ ๊ฒ์ ๋๋ค. Go์ธ์ด์์ ๋ฐํ ๊ฐ์ผ๋ก ์ฌ์ฉํ๋(T, error)
ํจํด๊ณผ ๋น์ทํ๊ธฐ ๋๋ฌธ์ ๋๋ค.
ํ ์คํธ
๋ฌ์คํธ์ ์นด๊ณ (cargo)๋ ๊ฐ๋จํ ๋จ์ ํ ์คํธ ํ๋ ์์ํฌ์ ํจ๊ป ์ ๊ณต๋ฉ๋๋ค:
-
๋จ์ ํ ์คํธ๋ ์ฝ๋ ์ ๋ฐ์์ ์ง์๋ฉ๋๋ค.
-
ํตํฉ ํ ์คํธ๋
tests/
๋๋ ํฐ๋ฆฌ๋ฅผ ํตํด ์ง์๋ฉ๋๋ค.
๋จ์ ํ ์คํธ
๋จ์ ํ
์คํธ๋ #[test]
๋ก ํ์ํฉ๋๋ค:
fn first_word(text: &str) -> &str {
match text.find(' ') {
Some(idx) => &text[..idx],
None => &text,
}
}
#[test]
fn test_empty() {
assert_eq!(first_word(""), "");
}
#[test]
fn test_single_word() {
assert_eq!(first_word("Hello"), "Hello");
}
#[test]
fn test_multiple_words() {
assert_eq!(first_word("Hello World"), "Hello");
}
cargo test
์ปค๋งจ๋๋ฅผ ์ฌ์ฉํ๋ฉด ๋จ์ ํ
์คํธ๋ฅผ ์ฐพ์์ ์คํํฉ๋๋ค.
ํ ์คํธ ๋ชจ๋
๋จ์ ํ ์คํธ๋ ์๋ ๋ชจ๋ ๋ฐ์ ์๋ธ ๋ชจ๋๋ก ๋ง๋๋ ๊ฒฝ์ฐ๊ฐ ๋ง์ต๋๋ค. (ํ๋ ์ด๊ทธ๋ผ์ด๋์์ ๋ค์ ํ ์คํธ๋ฅผ ์ํํด ๋ณด์ธ์):
fn helper(a: &str, b: &str) -> String { format!("{a} {b}") } pub fn main() { println!("{}", helper("Hello", "World")); } #[cfg(test)] mod tests { use super::*; #[test] fn test_helper() { assert_eq!(helper("foo", "bar"), "foo bar"); } }
- ์ด๋ ๊ฒ ์๋ธ ๋ชจ๋๋ก ํ ์คํธ๋ฅผ ๋ง๋ค๋ฉด privateํ ํฌํผ ํจ์์ ๋ํ ๋จ์ ํ ์คํธ๋ ๊ฐ๋ฅํฉ๋๋ค.
#[cfg(test)]
์ดํธ๋ฆฌ๋ทฐํธ๊ฐ ์ถ๊ฐ๋ ํญ๋ชฉ์cargo test
๋ฅผ ์ํํ์ ๊ฒฝ์ฐ์๋ง ๋์ํฉ๋๋ค.
๋ฌธ์ํ์ฃผ์ ํ ์คํธ
๋ฌ์คํธ๋ ๋ฌธ์ํ์ฃผ์์ ๋ํ ํ ์คํธ๋ฅผ ๋ด์ฅํ์ฌ ์ ๊ณตํฉ๋๋ค:
#![allow(unused)] fn main() { /// Shortens a string to the given length. /// /// ``` /// use playground::shorten_string; /// assert_eq!(shorten_string("Hello World", 5), "Hello"); /// assert_eq!(shorten_string("Hello World", 20), "Hello World"); /// ``` pub fn shorten_string(s: &str, length: usize) -> &str { &s[..std::cmp::min(length, s.len())] } }
///
์ฃผ์์์ ์ฝ๋ ๋ธ๋ก์ ์๋์ผ๋ก ๋ฌ์คํธ ์ฝ๋๋ก ์ธ์๋ฉ๋๋ค.- ์ด ์ฝ๋ ๋ธ๋ก์
cargo test
ํธ์ถํ๋ฉด ์๋์ผ๋ก ์ปดํ์ผ๋๊ณ ์คํ๋ฉ๋๋ค. - ์ ์ฝ๋๋ฅผ Rust Playground์์ ํ ์คํธ ํด ๋ณด์๊ธฐ ๋ฐ๋๋๋ค.
ํตํฉ ํ ์คํธ
๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ฌ์ฉ์ ์ ์ฅ์์ ํ ์คํธ ํ๋ ค๋ฉด, ํตํฉ ํ ์คํธ๋ฅผ ํด์ผ ํฉ๋๋ค.
test/
๋๋ ํฐ๋ฆฌ ์๋์ .rs
ํ์ผ์ ํ๋ ๋ง๋์ธ์:
use my_library::init;
#[test]
fn test_init() {
assert!(init().is_ok());
}
์ด ํ ์คํธ๋ ํฌ๋ ์ดํธ์ ๊ณต๊ฐ API์๋ง ์ ๊ทผํ ์ ์์ต๋๋ค.
ํ ์คํธ ์์ฑ์ ์ ์ฉํ ํฌ๋ ์ดํธ
๋ฌ์คํธ๋ ํ ์คํธ ์์ฑ๊ณผ ๊ด๋ จํด์๋ ๊ธฐ๋ณธ์ ์ธ ๊ธฐ๋ฅ๋ง ์ง์ํฉ๋๋ค.
๋ค์์ ํ ์คํธ๋ฅผ ์์ฑํ ๋ ๊ถ์ฅ๋๋ ์ถ๊ฐ ํฌ๋ ์ดํธ์ ๋๋ค:
- googletest: C++์ฉ GoogleTest์ ๋ฐฉ์์ ๋ฐ๋ฅด๋ ํ ์คํธ ๋ผ์ด๋ธ๋ฌ๋ฆฌ ์ ๋๋ค.
- proptest: Rust์ฉ ์์ฑ ๊ธฐ๋ฐ ํ ์คํธ์ ๋๋ค.
- rstest: ํฝ์ค์ฒ ๋ฐ ๋งค๊ฐ๋ณ์ํ๋ ํ ์คํธ๋ฅผ ์ง์ํฉ๋๋ค.
์์ ํ์ง ์์ ๋ฌ์คํธ
๋ฌ์คํธ๋ก ์์ฑ๋ ํ๋ก๊ทธ๋จ์ ํฌ๊ฒ ๋ ๋ถ๋ถ์ผ๋ก ๋๋ฉ๋๋ค:
- ์์ ํ ๋ฌ์คํธ: ๋ฉ๋ชจ๋ฆฌ ๊ด๋ จ ์ค๋ฅ ๋ฐ์ ๋ถ๊ฐ๋ฅ, ์ ์๋์ง ์์ ๋์ ์์.
- ์์ ํ์ง ์์ ๋ฌ์คํธ: ํน๋ณํ ์กฐ๊ฑด์ ๋ง์กฑํ์ง ์์์ฑ๋ก ์ฌ์ฉ๋๋ฉด ์ ์๋์ง ์์ ๋์์ ์ ๋ฐํ ์ ์์.
์ด ๊ฐ์๋ ๋๋ถ๋ถ ์์ ํ ๋ฌ์คํธ์ ๋ํด ๋ค๋ฃจ์ง๋ง ์์ ํ์ง ์์ ๋ฌ์คํธ๊ฐ ๋ฌด์์ธ์ง๋ ์์ ๋์ด์ผ ํฉ๋๋ค.
๋ณดํต, ์์ ํ์ง ์์ ๋ฌ์คํธ ์ฝ๋๋ ํฌ๊ธฐ๊ฐ ์์ผ๋ฉฐ, ๋ ๋ฆฝ์ ์ผ๋ก ์กด์ฌํฉ๋๋ค. ๊ทธ๋ฆฌ๊ณ ์ฝ๋๊ฐ ์ ์ ์๋ํ๋์ง์ ๋ํด ์ธ๋ฐํ๊ฒ ๋ฌธ์ํ๊ฐ ๋์ด ์์ต๋๋ค. ๊ทธ๋ฆฌ๊ณ , ๋ง์ ๊ฒฝ์ฐ ์์ ํ ๋ฌ์คํธ ์ฝ๋๋ฅผ ํตํด์ ์ถ์ํ๋ฅผ ์ํจ ํ ์ฌ์ฉํ๊ฒ ๋ฉ๋๋ค.
์์ ํ์ง ์์ ๋ฌ์คํธ๋ฅผ ์ด์ฉํ๋ฉด ๋ค์๊ณผ ๊ฐ์ ๋ค์ฏ ๊ฐ์ง ๊ฒ๋ค์ด ๊ฐ๋ฅํด ์ง๋๋ค:
- ์์ ํฌ์ธํฐ ์ญ์ฐธ์กฐ(๋ฐ๋ผ๊ฐ๊ธฐ)
- ์ ์ ๊ฐ๋ณ๋ณ์ ์ ๊ทผ ๋ฐ ์์ .
union
ํ๋ ์ ๊ทผ.extern
ํจ์๋ฅผ ํฌํจํunsafe
ํจ์ ํธ์ถ.unsafe
ํธ๋ ์ ๊ตฌํ.
์ ๊ธฐ๋ฅ๋ค์ ๋ํด ๊ฐ๋ตํ ์ดํด๋ณด๊ฒ ์ต๋๋ค. ์์ธํ ๋ด์ฉ์ ๋ฌ์คํธ ํ๋ก๊ทธ๋๋ฐ ์ธ์ด, 19.1์ ๊ณผ Rustonomicon๋ฅผ ์ฐธ์กฐํ์ธ์.
์์ ํ์ง ์์ ๋ฌ์คํธ๋ผ๊ณ ํด์ ์ฝ๋๊ฐ ๋ถ์ ํ ํ๋ค๋ ๋ป์ ์๋๋๋ค. ์ฌ๊ธฐ์ ์์ ํ์ง ์๋ค์ ์๋ฏธ๋ ์ปดํ์ผ๋ฌ๊ฐ ์ ๊ณตํด์ฃผ๋ ์์ ์ฅ์น๋ค์ด ๊บผ์ง ์ํ์ด๋ฉฐ, ๊ฐ๋ฐ์๊ฐ ์ค์ค๋ก ์ ํํ๊ณ ์์ ํ ์ฝ๋๋ฅผ ์์ฑํด์ผ ํจ์ ์๋ฏธํฉ๋๋ค. ์ด๋ ์ปดํ์ผ๋ฌ๊ฐ ๋ ์ด์ ๋ฌ์คํธ์ ๋ฉ๋ชจ๋ฆฌ ์์ ๊ณผ ๊ด๋ จ๋ ๊ท์น๋ค์ ์ ์ฉํ์ง ์๋๋ค๋ ๊ฒ์ ๋๋ค.
์์ ํฌ์ธํฐ ์ญ์ฐธ์กฐ(๋ฐ๋ผ๊ฐ๊ธฐ)
ํฌ์ธํฐ๋ฅผ ๋ง๋๋ ๊ฒ์ ์์ ํฉ๋๋ค. ํ์ง๋ง ์ญ์ฐธ์กฐ(๋ฐ๋ผ๊ฐ๊ธฐ)ํ ๊ฒฝ์ฐ unsafe
๊ฐ ํ์ํฉ๋๋ค:
fn main() { let mut num = 5; let r1 = &mut num as *mut i32; let r2 = r1 as *const i32; // Safe because r1 and r2 were obtained from references and so are // guaranteed to be non-null and properly aligned, the objects underlying // the references from which they were obtained are live throughout the // whole unsafe block, and they are not accessed either through the // references or concurrently through any other pointers. unsafe { println!("r1 is: {}", *r1); *r1 = 10; println!("r2 is: {}", *r2); } }
๋ชจ๋ unsafe
๋ธ๋ก์ ๋ํด ์ ๊ทธ ์ฝ๋๊ฐ ์์ ํ์ง์ ๋ํ ์ค๋ช
์ ์ฃผ์์ผ๋ก ๋ค๋ ๊ฒ์ ์ข์ ์ต๊ด์
๋๋ค(์ฌ์ค ์๋๋ก์ด๋์ ๋ฌ์คํธ ์คํ์ผ ๊ฐ์ด๋์์๋ ์ด๊ฒ ํ์์
๋๋ค).
ํฌ์ธํฐ ์ญ์ฐธ์กฐ๋ฅผ ํ ๊ฒฝ์ฐ, ํฌ์ธํฐ๊ฐ ์ ํจํด์ผ ํฉ๋๋ค. ์๋ฅผ ๋ค์ด:
- ํฌ์ธํฐ๋ null์ด๋ฉด ์๋ฉ๋๋ค.
- ํฌ์ธํฐ๋ ๋ฐ๋ผ๊ฐ๊ธฐ๊ฐ ๊ฐ๋ฅํด์ผ ํฉ๋๋ค (๊ฐ์ฒด์ ์ด๋ ํ ๋ถ๋ถ์ ๊ฐ๋ฆฌํค๊ณ ์์ด์ผ ํฉ๋๋ค).
- ์ด๋ฏธ ๋ฐํ๋ ๊ฐ์ฒด๋ฅผ ๊ฐ๋ฆฌํค๋ฉด ์๋ฉ๋๋ค.
- ๊ฐ์ ์์น์ ๋ํด ๋์์ ์ธ ์ ๊ทผ์ด ์์ผ๋ฉด ์๋ฉ๋๋ค.
- ์ฐธ์กฐ๋ฅผ ์บ์คํ ํด์ ํฌ์ธํฐ๋ฅผ ๋ง๋ค์๋ค๋ฉด, ๊ทธ ์ฐธ์กฐ๊ฐ ๊ฐ๋ฆฌํค๋ ๊ฐ์ฒด๋ ์ด์ ์์ด์ผ ํ๋ฉฐ, ๊ทธ ๊ฐ์ฒด์ ๋ฉ๋ชจ๋ฆฌ๋ฅผ ์ ๊ทผํ๋ ์ฐธ์กฐ๊ฐ ํ๋๋ ์์ด์ผ ํฉ๋๋ค.
๋๋ถ๋ถ์ ๊ฒฝ์ฐ ํฌ์ธํฐ๋ align๋์ด ์์ด์ผ ํฉ๋๋ค.
์ ์ ๊ฐ๋ณ ๋ณ์
๋ถ๋ณ ์ ์ ๋ณ์๋ฅผ ์ฝ๋ ๊ฒ์ ์์ ํฉ๋๋ค:
static HELLO_WORLD: &str = "Hello, world!"; fn main() { println!("HELLO_WORLD: {HELLO_WORLD}"); }
ํ์ง๋ง, ๋ฐ์ดํฐ ๋ ์ด์ค๊ฐ ๋ฐ์ํ ์ ์์ผ๋ฏ๋ก ์ ์ ๊ฐ๋ณ๋ณ์๋ฅผ ์ฝ๊ณ ์ฐ๋ ๊ฒ์ ์์ ํ์ง ์์ต๋๋ค:
static mut COUNTER: u32 = 0; fn add_to_counter(inc: u32) { unsafe { COUNTER += inc; } // Potential data race! } fn main() { add_to_counter(42); unsafe { println!("COUNTER: {COUNTER}"); } // Potential data race! }
์ผ๋ฐ์ ์ผ๋ก ์ด์ผ๊ธฐ ํด์, ์ ์ ๊ฐ๋ณ ๋ณ์๋ฅผ ์ฐ๋ ๊ฒ์ ์ข์ ์์ด๋์ด๊ฐ ์๋๋๋ค. ๊ทธ๋ฌ๋ no_std
์ ๊ฐ์ ์ ์์ค ์ฝ๋ฉ์ ํ ๊ฒฝ์ฐ์๋ ํ์ํ๊ธฐ๋ ํฉ๋๋ค. ์๋ฅผ ๋ค์ด ํ ํ ๋น๊ธฐ๋ฅผ ๊ตฌํํ๊ฑฐ๋, C API๋ฅผ ์ฌ์ฉํ๋ ๊ฒ ๊ทธ๋ฐ ๊ฒฝ์ฐ์
๋๋ค.
Unions
์ ๋์จ ํ์ ์ ์ด๊ฑฐํ(enum)๊ณผ ๋น์ทํ์ง๋ง, ์ด๋ค ํ๋์ ํด๋นํ๋ ๊ฐ์ ๊ฐ์ง๊ณ ์๋์ง ์ฌ๋ถ๋ฅผ ํ๋ก๊ทธ๋๋จธ๊ฐ ์๋์ผ๋ก ์ถ์ ํด์ผ ํฉ๋๋ค:
#[repr(C)] union MyUnion { i: u8, b: bool, } fn main() { let u = MyUnion { i: 42 }; println!("int: {}", unsafe { u.i }); println!("bool: {}", unsafe { u.b }); // Undefined behavior! }
๋ฌ์คํธ์๋ ์ด๊ฑฐํ์ด ์๊ธฐ ๋๋ฌธ์ ์ ๋์จ์ด ํ์ํ ๊ฒฝ์ฐ๋ ๊ทนํ ๋๋ญ ๋๋ค. ์ ๋์จ์ C ๋ผ์ด๋ธ๋ฌ๋ฆฌ API๋ฅผ ์ฌ์ฉํ ๋ ๊ฐ๋ ํ์ํฉ๋๋ค.
๋ฐ์ดํธ๋ค์ ํน์ ํ์
์ผ๋ก ์ฌํด์ ํ๊ณ ์ถ๋ค๋ฉด std::mem::transmute
๋ ์ข ๋ ์์ ํ zerocopy
ํฌ๋ ์ดํธ๋ฅผ ์ฌ์ฉํ์ธ์.
์์ ํ์ง ์์ ํจ์ ํธ์ถ
ํจ์๋ ๋ฉ์๋๊ฐ ์ ์๋์ง ์์ ๋์์ผ๋ก ๋น ์ง์ง ์๊ฒ ํ๊ธฐ ์ํด์ ๋ง์กฑํด์ผ ํ๋ ์ ์ ์กฐ๊ฑด์ด ์๋ ๊ฒฝ์ฐ, ๊ทธ ํจ์๋ ๋ฉ์๋๋ฅผ unsafe
๋ก ํ์ํ ์ ์์ต๋๋ค:
fn main() { let emojis = "๐ปโ๐"; // Safe because the indices are in the correct order, within the bounds of // the string slice, and lie on UTF-8 sequence boundaries. unsafe { println!("emoji: {}", emojis.get_unchecked(0..4)); println!("emoji: {}", emojis.get_unchecked(4..7)); println!("emoji: {}", emojis.get_unchecked(7..11)); } println!("char count: {}", count_chars(unsafe { emojis.get_unchecked(0..7) })); // Not upholding the UTF-8 encoding requirement breaks memory safety! // println!("emoji: {}", unsafe { emojis.get_unchecked(0..3) }); // println!("char count: {}", count_chars(unsafe { emojis.get_unchecked(0..3) })); } fn count_chars(s: &str) -> usize { s.chars().map(|_| 1).sum() }
์์ ํ์ง ์์ ํจ์ ์์ฑํ๊ธฐ
์ฌ๋ฌ๋ถ์ด ์์ฑํ ํจ์๋ฅผ ์ฌ์ฉํ ๋ ์ด๋ค ํน๋ณํ ์กฐ๊ฑด์ ๋ง์กฑํด์ผ ํ๋ค๋ฉด, unsafe
๋ก ๋งํนํ ์ ์์ต๋๋ค.
/// Swaps the values pointed to by the given pointers. /// /// # Safety /// /// The pointers must be valid and properly aligned. unsafe fn swap(a: *mut u8, b: *mut u8) { let temp = *a; *a = *b; *b = temp; } fn main() { let mut a = 42; let mut b = 66; // Safe because ... unsafe { swap(&mut a, &mut b); } println!("a = {}, b = {}", a, b); }
์ฐธ์กฐ๋ฅผ ์ฌ์ฉํ๋ฉด ์์ ํ๊ฒ ๊ตฌํํ ์ ์๊ธฐ ๋๋ฌธ์, ์ค์ ๋ก ํฌ์ธํฐ๋ฅผ ์ฌ์ฉํ ํ์๋ ์์ต๋๋ค.
์์ ํ์ง ์์(unsafe) ์ฝ๋๊ฐ ์์ ํ์ง ์์(unsafe) ํจ์์ ๋ด๋ถ์์ ํธ์ถ๋ ๊ฒฝ์ฐ์๋ unsafe
๋ธ๋ก์ ์ง์ ํ์ง ์์๋ ๋ฉ๋๋ค. unsafe
๋ธ๋ก์ ํญ์ ์ง์ ํ๋๋ก ํ๊ณ ์ถ๋ค๋ฉด #[deny(unsafe_op_in_unsafe_fn)]
๋ฅผ ์ด์ฉํ์ธ์. ์ด ์ดํธ๋ฆฌ๋ทฐํธ๋ฅผ ์ถ๊ฐํด ๋ณด๊ณ ์ด๋ค ์ผ์ด ์ผ์ด๋๋์ง ํ์ธํด ๋ณด์ธ์.
์ธ๋ถ ์ฝ๋ ํธ์ถ
๋ค๋ฅธ ์ธ์ด์ ํจ์๋ ๋ฌ์คํธ์ ๋ณด์ฆ์ ์๋ฐํ ์ ์์ต๋๋ค. ๋ฐ๋ผ์ ์ด๋ฅผ ํธ์ถํ๋ ๊ฒ์ ์์ ํ์ง ์์ต๋๋ค:
extern "C" { fn abs(input: i32) -> i32; } fn main() { unsafe { // Undefined behavior if abs misbehaves. println!("Absolute value of -3 according to C: {}", abs(-3)); } }
์ด๊ฒ ๋ฌธ์ ๊ฐ ๋๋ ๊ฒฝ์ฐ๋ ๋๋ถ๋ถ ์ธ๋ถ ํจ์๊ฐ ๋ฌ์คํธ์ ๋ฉ๋ชจ๋ฆฌ ๋ชจ๋ธ์ ์๋ฐํ๊ณ ์์ ๊ฒฝ์ฐ์ ๋๋ค. ๊ทธ๋ฌ๋ ์ด๋ค Cํจ์๋ผ๋ ์ด๋ค ์์์ ์ํฉ์์๋ ์ ์๋์ง ์์ ๋์์ ํ ์ ์๊ธฐ ๋๋ฌธ์, ์๋ฐํ ๋งํด์๋ ๋ชจ๋ ์ธ๋ถ ํจ์์ ๋ํด์ ๋ฌธ์ ์ ๋๋ค.
์ ์์ ์ฝ๋์์ "C"
๋ ABI๋ฅผ ์๋ฏธํฉ๋๋ค. ๋ค๋ฅธ ABI๋ ์์ต๋๋ค.
์์ ํ์ง ์์ ํธ๋ ์ ๊ตฌํํ๊ธฐ
ํจ์์์์ ๋ง์ฐฌ๊ฐ์ง๋ก ํธ๋ ์๋ unsafe
๋ก ๋งํน ๊ฐ๋ฅํฉ๋๋ค. ๋ง์ฝ ๊ทธ ํธ๋ ์์ ๊ตฌํํ ๋ ์ ์๋์ง ์์ ๋์์ ํผํ๊ธฐ ์ํด ํน๋ณํ ์กฐ๊ฑด์ด ํ์ํ๋ค๋ฉด ๋ง์ด์ง์.
์๋ฅผ ๋ค์ด zerocopy
ํฌ๋ ์ดํธ์๋ ์์ ํ์ง ์์ ํธ๋ ์์ด ์์ต๋๋ค:
use std::mem::size_of_val; use std::slice; /// ... /// # Safety /// The type must have a defined representation and no padding. pub unsafe trait AsBytes { fn as_bytes(&self) -> &[u8] { unsafe { slice::from_raw_parts(self as *const Self as *const u8, size_of_val(self)) } } } // Safe because u32 has a defined representation and no padding. unsafe impl AsBytes for u32 {}
์์ ํ์ง ์์ ํธ๋ ์์ ๋ง๋ค ๋์๋ ์ฃผ์์ # Safety
ํญ๋ชฉ์ด ์์ด์ ์ด ํธ๋ ์์ ์์ ํ๊ฒ ๊ตฌํํ๋ ค๋ฉด ์ด๋ค ์๊ตฌ์ฌํญ๋ค์ ๋ง์กฑํด์ผ ํ๋์ง๋ฅผ ์ค๋ช
ํด์ผ ํฉ๋๋ค.
AsBytes
์์ ์ง์ผ์ผ ํ ์์ ์ฑ์ ๋ํ ์ค์ ์ค๋ช
์ ์ข ๋ ๊ธธ๊ณ ๋ณต์กํฉ๋๋ค.
๋นํธ์ธ ํธ๋ ์์ธ Send
์ Sync
๋ ์์ ํ์ง ์์ ํธ๋ ์ ์
๋๋ค.
3์ผ์ฐจ ์คํ ์ฐ์ต๋ฌธ์
๋๋ ํฐ๋ฆฌ์ ๋ด์ฉ์ ์ฝ๋ ์์ ํ ๋ํผ ์ฝ๋๋ฅผ ์์ฑํด ๋ด ์๋ค!
์ด๋ฒ ์ฐ์ต์์๋, ํ๋ ์ด๊ทธ๋ผ์ด๋ ๋์ ๋ก์ปฌ ๊ฐ๋ฐ ํ๊ฒฝ์ ์ฌ์ฉํด์, ์ฌ๋ฌ๋ถ์ ์ปดํจํฐ์์ ๋ฐ์ด๋๋ฆฌ๋ฅผ ์ง์ ์ํํด ๋ณด์ธ์.
์์ํ๊ธฐ ์ํด, ๋ก์ปฌ ํ๊ฒฝ์์ ์ํํ๊ธฐ๋ฅผ ๋ฐ๋ฅด์ธ์.
์ฐ์ต๋ฌธ์ ๋ฅผ ์ดํด ๋ณธ ํ, ์ ๊ณต๋ ํด๋ต์ ์ดํด๋ณผ ์ ์์ต๋๋ค.
FFI๋ํผ
๋ฌ์คํธ๋ _์ธ๋ถ ๊ธฐ๋ฅ ํธ์ถ(FFI)_์ ์ง์ํฉ๋๋ค. ์ฐ๋ฆฌ๋ ์ด๋ฅผ ์ด์ฉํ์ฌ ๋๋ ํฐ๋ฆฌ์์ ํ์ผ ์ด๋ฆ์ ์ฝ์ด์ค๋ libc
ํจ์์ ๋ํ ์์ ํ ๋ํผ๋ฅผ ๋ง๋ค ๊ฒ์
๋๋ค.
์๋ ๋ฆฌ๋ ์ค ๋ฉ๋ด์ผ ๋ฌธ์๋ค์ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค:
์๋ง std::ffi
๋ชจ๋์ ์ฐธ์กฐํ ํ์๊ฐ ์์ ๊ฒ์
๋๋ค. ๊ฑฐ๊ธฐ์๋ ์ด๋ฒ ์์ ๋ฅผ ์ํํ๋๋ฐ ํ์ํ ๋ค์ํ ์ข
๋ฅ์ ๋ฌธ์์ด ํ์
๋ค์ด ์๊ฐ๋์ด ์์ต๋๋ค:
ํ์ | ์ธ์ฝ๋ฉ | ์ฌ์ฉ |
---|---|---|
str ๊ณผ String | UTF-8 | ๋ฌ์คํธ์์์ ๋ฌธ์์ด ์ฒ๋ฆฌ |
CStr ๊ณผ CString | ๋(NUL)๋ก ๋๋จ | Cํจ์์ ์ฐ๋ํ๊ธฐ |
OsStr ์ OsString | OS๊ฐ ์ ์ํจ | OS์ ์ฐ๋ํ๊ธฐ ์ํ ๋ฌธ์์ด |
์ด ํ์ ๋ค ๊ฐ์ ๋ณํ์ ๋ค์๊ณผ ๊ฐ์ต๋๋ค:
&str
์์CString
์ผ๋ก์ ๋ณํ: ๋งจ ๋ง์ง๋ง์\0
๋ฌธ์๋ฅผ ์ ์ฅํ๊ธฐ ์ํ ๊ณต๊ฐ์ ํ ๋นํด์ผ ํฉ๋๋ค,CString
์์*const i8
๋ก์ ๋ณํ: Cํจ์๋ฅผ ํธ์ถํ๊ธฐ ์ํด์๋ ํฌ์ธํฐ๊ฐ ํ์ํฉ๋๋ค,*const i8
์์&CStr
๋ก์ ๋ณํ: ์ฃผ์ด์ง ๋ฐ์ดํธ ์ํ์ค๊ฐ\0
๋ก ๋๋๋์ง ํ์ธํ๊ณ ์ถ์ ๊ฒฝ์ฐ,&CStr
์์&[u8]
๋ก์ ๋ณํ: ๋ฐ์ดํธ ์ฌ๋ผ์ด์ค๋ โ์์์๋ ๋ฐ์ดํฐโ์ ๋ํ ์ผ๋ฐ์ ์ธ ์ธํฐํ์ด์ค์ ๋๋ค,&[u8]
์์&OsStr
๋ก์ ๋ณํ:&OsStr
๋OsString
์ผ๋ก ๊ฐ๊ธฐ ์ํ ์ค๊ฐ ๋จ๊ณ ์ ๋๋ค.OsStrExt
๋ฅผ ์ฌ์ฉํด์OsStr
๋ฅผ ์์ฑํ์ธ์,&OsStr
์์OsString
์ผ๋ก์ ๋ณํ:&OsStr
์ด ๊ฐ๋ฆฌํค๊ณ ์๋ ๋ฐ์ดํฐ๋ฅผ ๋ณต์ฌํจ์ผ๋ก์จ, ์ด ๋ฐ์ดํฐ๋ฅผ ๋ฆฌํดํ๊ณ ,readdir
ํจ์๋ฅผ ํธ์ถํ ๋ ์ฌ์ฉํ ์ ์๊ฒ ํด ์ค๋๋ค.
Nomicon์ FFI์ ๊ด๋ จํ ์์ฃผ ์ ์ฉํ ์ฑํฐ๊ฐ ์์ต๋๋ค.
์๋ ์ฝ๋๋ฅผ https://play.rust-lang.org/์ ๋ณต์ฌํ๊ณ ๋น ์ง ํจ์์ ๋ฉ์๋๋ฅผ ์ฑ์๋ด ๋๋ค:
// TODO: remove this when you're done with your implementation. #![allow(unused_imports, unused_variables, dead_code)] mod ffi { use std::os::raw::{c_char, c_int}; #[cfg(not(target_os = "macos"))] use std::os::raw::{c_long, c_ulong, c_ushort, c_uchar}; // Opaque type. See https://doc.rust-lang.org/nomicon/ffi.html. #[repr(C)] pub struct DIR { _data: [u8; 0], _marker: core::marker::PhantomData<(*mut u8, core::marker::PhantomPinned)>, } // Layout according to the Linux man page for readdir(3), where ino_t and // off_t are resolved according to the definitions in // /usr/include/x86_64-linux-gnu/{sys/types.h, bits/typesizes.h}. #[cfg(not(target_os = "macos"))] #[repr(C)] pub struct dirent { pub d_ino: c_ulong, pub d_off: c_long, pub d_reclen: c_ushort, pub d_type: c_uchar, pub d_name: [c_char; 256], } // Layout according to the macOS man page for dir(5). #[cfg(all(target_os = "macos"))] #[repr(C)] pub struct dirent { pub d_fileno: u64, pub d_seekoff: u64, pub d_reclen: u16, pub d_namlen: u16, pub d_type: u8, pub d_name: [c_char; 1024], } extern "C" { pub fn opendir(s: *const c_char) -> *mut DIR; #[cfg(not(all(target_os = "macos", target_arch = "x86_64")))] pub fn readdir(s: *mut DIR) -> *const dirent; // See https://github.com/rust-lang/libc/issues/414 and the section on // _DARWIN_FEATURE_64_BIT_INODE in the macOS man page for stat(2). // // "Platforms that existed before these updates were available" refers // to macOS (as opposed to iOS / wearOS / etc.) on Intel and PowerPC. #[cfg(all(target_os = "macos", target_arch = "x86_64"))] #[link_name = "readdir$INODE64"] pub fn readdir(s: *mut DIR) -> *const dirent; pub fn closedir(s: *mut DIR) -> c_int; } } use std::ffi::{CStr, CString, OsStr, OsString}; use std::os::unix::ffi::OsStrExt; #[derive(Debug)] struct DirectoryIterator { path: CString, dir: *mut ffi::DIR, } impl DirectoryIterator { fn new(path: &str) -> Result<DirectoryIterator, String> { // Call opendir and return a Ok value if that worked, // otherwise return Err with a message. unimplemented!() } } impl Iterator for DirectoryIterator { type Item = OsString; fn next(&mut self) -> Option<OsString> { // Keep calling readdir until we get a NULL pointer back. unimplemented!() } } impl Drop for DirectoryIterator { fn drop(&mut self) { // Call closedir as needed. unimplemented!() } } fn main() -> Result<(), String> { let iter = DirectoryIterator::new(".")?; println!("files: {:#?}", iter.collect::<Vec<_>>()); Ok(()) }
Welcome to Rust in Android
๋ฌ์คํธ๋ ์๋๋ก์ด๋ ๋ค์ดํฐ๋ธ ํ๋ซํผ ๊ฐ๋ฐ์ ์ง์ํฉ๋๋ค. ๊ธฐ์กด์ OS ์๋น์ค๋ฅผ ํ์ฅํ๊ฑฐ๋, ์๋ก์ด ์๋น์ค๋ฅผ ๋ง๋๋๋ฐ ๋ฌ์คํธ๋ฅผ ์ธ ์ ์์ต๋๋ค.
์ฐ๋ฆฌ๋ ์ค๋ ์ฌ๋ฌ๋ถ์ ํ๋ก์ ํธ์์ ๋ฌ์คํธ ์ฝ๋๋ฅผ ํธ์ถํด๋ณผ ๊ฒ์ ๋๋ค. ๊ทธ ํ๋ก์ ํธ์์ ๋ฌ์คํธ๋ก ์ฎ๊ธธ๋ง ํ ์์ ๋ถ๋ถ์ ์ ํ์ธ์. ์์กด์ฑ์ด ์ ๊ณ โํน์ดํโ ํ์ ์ด ์ ์ ์๋ก ์ข์ต๋๋ค. ๋ฐ์ดํธ ๋ช ๊ฐ๋ฅผ ํ์ฑํ๋ ์ฝ๋๋ผ๋ฉด ์๋ฒฝํฉ๋๋ค.
์ค์น
์๋๋ก์ด๋ ๊ฐ์ ๋๋ฐ์ด์ค(Android Virtual Device)๋ฅผ ์ฌ์ฉํ์ฌ ์ฌ๋ฌ๋ถ์ ์ฝ๋๋ฅผ ์ํํ ๊ฒ๋๋ค. ์๋ก์ด ๊ฐ์ ๋๋ฐ์ด์ค๋ฅผ ์์ฑํ๋ ค๋ฉด ์๋์ ๋ช ๋ น์ด๋ฅผ ์ฌ์ฉํ์ธ์:
source build/envsetup.sh
lunch aosp_cf_x86_64_phone-userdebug
acloud create
์์ธํ ๋ด์ฉ์ Android Developer Codelab์ ์ฐธ์กฐํ์ญ์์ค.
๋น๋ ๊ท์น
์๋๋ก์ด๋ ๋น๋ ์์คํ (Soong)์ ๋ค์๊ณผ ๊ฐ์ ์ฌ๋ฌ ๋ชจ๋์ ํตํด ๋ฌ์คํธ๋ฅผ ์ง์ํฉ๋๋ค:
Module Type | Description |
---|---|
rust_binary | ๋ฌ์คํธ ๋ฐ์ด๋๋ฆฌ๋ฅผ ์์ฑํฉ๋๋ค. |
rust_library | ๋ฌ์คํธ ๋ผ์ด๋ธ๋ฌ๋ฆฌ(rlibํน์ dylib)๋ฅผ ์์ฑํฉ๋๋ค. |
rust_ffi | cc ๋ชจ๋์์ ์ฌ์ฉํ ์ ์๋ C library (์ ์ ํน์ ๋์ )๋ฅผ ์์ฑํฉ๋๋ค. |
rust_proc_macro | proc-macro ๋ฅผ ๊ตฌํํ๋ ๋ฌ์คํธ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์์ฑํฉ๋๋ค. ์ปดํ์ผ๋ฌ์ ํ๋ฌ๊ทธ์ธ์ผ๋ก ์๊ฐํด๋ ์ข์ต๋๋ค. |
rust_test | ํ์ค ๋ฌ์คํธ ํ ์คํธ ๋ฌ๋๋ฅผ ์ฌ์ฉํ๋ ํ ์คํธ ๋ฐ์ด๋๋ฆฌ๋ฅผ ์์ฑํฉ๋๋ค. |
rust_fuzz | libfuzzer ๋ฅผ ์ฌ์ฉํ์ฌ fuzz ๋ฐ์ด๋๋ฆฌ๋ฅผ ์์ฑํฉ๋๋ค. |
rust_protobuf | ํ๋กํ ๋ฒํ(protobuf) ์ธํฐํ์ด์ค๋ฅผ ์ ๊ณตํ๋ ๋ฌ์คํธ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์์ฑํฉ๋๋ค. |
rust_bindgen | C ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ๋ํ ๋ฌ์คํธ ๋ฐ์ธ๋ฉ์ ์ ๊ณตํ๋ ๋ฌ์คํธ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์์ฑํฉ๋๋ค. |
๋ค์์ rust_binary
์ rust_library
๋ฅผ ์ดํด๋ด
๋๋ค.
๋ฌ์คํธ ๋ฐ์ด๋๋ฆฌ
๊ฐ๋จํ ์์ฉ ํ๋ก๊ทธ๋จ์ผ๋ก ์์ํด ๋ณด๊ฒ ์ต๋๋ค. AOSP ์ฒดํฌ์์์ ๋ฃจํธ์์ ๋ค์ ํ์ผ์ ์์ฑํฉ๋๋ค:
hello_rust/Android.bp:
rust_binary {
name: "hello_rust",
crate_name: "hello_rust",
srcs: ["src/main.rs"],
}
hello_rust/src/main.rs:
//! Rust demo. /// Prints a greeting to standard output. fn main() { println!("Hello from Rust!"); }
๊ทธ๋ฐ ๋ค์, ์ด ๋ฐ์ด๋๋ฆฌ๋ฅผ ๋น๋ํ๊ณ , ๊ฐ์ ๋๋ฐ์ด์ค์ ๋ฃ๊ณ , ์คํํฉ๋๋ค:
m hello_rust
adb push "$ANDROID_PRODUCT_OUT/system/bin/hello_rust /data/local/tmp"
adb shell /data/local/tmp/hello_rust
Hello from Rust!
๋ฌ์คํธ ๋ผ์ด๋ธ๋ฌ๋ฆฌ
rust_library
๋ฅผ ์ฌ์ฉํ์ฌ ์๋๋ก์ด๋์ฉ ์ ๋ฌ์คํธ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ๋ง๋ญ๋๋ค.
์ฌ๊ธฐ์ ๋ ๊ฐ์ ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ๋ํ ์์กด์ฑ์ ์ ์ธํฉ๋๋ค:
- ์๋์ ์ ์ํ
libgreeting
. external/rust/crates/
์ ์กด์ฌํ๋libtextwrap
.
hello_rust/Android.bp:
rust_binary {
name: "hello_rust_with_dep",
crate_name: "hello_rust_with_dep",
srcs: ["src/main.rs"],
rustlibs: [
"libgreetings",
"libtextwrap",
],
prefer_rlib: true,
}
rust_library {
name: "libgreetings",
crate_name: "greetings",
srcs: ["src/lib.rs"],
}
hello_rust/src/main.rs:
//! Rust demo.
use greetings::greeting;
use textwrap::fill;
/// Prints a greeting to standard output.
fn main() {
println!("{}", fill(&greeting("Bob"), 24));
}
hello_rust/src/lib.rs:
//! Greeting library.
/// Greet `name`.
pub fn greeting(name: &str) -> String {
format!("Hello {name}, it is very nice to meet you!")
}
์ด์ ์ฒ๋ผ, ๋น๋ํ๊ณ , ๊ฐ์ ๋๋ฐ์ด์ค๋ก ๋ฃ๊ณ , ์คํํฉ๋๋ค:
m hello_rust_with_dep
adb push "$ANDROID_PRODUCT_OUT/system/bin/hello_rust_with_dep /data/local/tmp"
adb shell /data/local/tmp/hello_rust_with_dep
Hello Bob, it is very
nice to meet you!
AIDL
๋ฌ์คํธ๋ ์๋๋ก์ด๋ ์ธํฐํ์ด์ค ์ ์ ์ธ์ด(AIDL)๋ฅผ ์ง์ํฉ๋๋ค:
- ๋ฌ์คํธ ์ฝ๋์์ ๊ธฐ์กด AIDL ์๋ฒ๋ฅผ ํธ์ถ ํ ์ ์์ต๋๋ค.
- ๋ฌ์คํธ์์ ์๋ก์ด AIDL ์๋ฒ๋ฅผ ์์ฑํ ์ ์์ต๋๋ค.
AIDL ์ธํฐํ์ด์ค
AIDL ์ธํฐํ์ด์ค๋ฅผ ์ด์ฉํด์ ์๋น์ค์ API๋ฅผ ์ ์ธํฉ๋๋ค:
birthday_service/aidl/com/example/birthdayservice/IBirthdayService.aidl:
package com.example.birthdayservice;
/** Birthday service interface. */
interface IBirthdayService {
/** Generate a Happy Birthday message. */
String wishHappyBirthday(String name, int years);
}
birthday_service/aidl/Android.bp:
aidl_interface {
name: "com.example.birthdayservice",
srcs: ["com/example/birthdayservice/*.aidl"],
unstable: true,
backend: {
rust: { // Rust is not enabled by default
enabled: true,
},
},
}
AIDL ํ์ผ์ด ๋ฒค๋ ํํฐ์
์ ์๋ ๋ฐ์ด๋๋ฆฌ์์ ์ฌ์ฉ๋ ๊ฒฝ์ฐ vendor_available: true
๋ฅผ ์ถ๊ฐํฉ๋๋ค.
์๋น์ค ๊ตฌํ
์ด์ AIDL์๋น์ค๋ฅผ ๊ตฌํํ ์ ์์ต๋๋ค:
birthday_service/src/lib.rs:
//! Implementation of the `IBirthdayService` AIDL interface.
use com_example_birthdayservice::aidl::com::example::birthdayservice::IBirthdayService::IBirthdayService;
use com_example_birthdayservice::binder;
/// The `IBirthdayService` implementation.
pub struct BirthdayService;
impl binder::Interface for BirthdayService {}
impl IBirthdayService for BirthdayService {
fn wishHappyBirthday(&self, name: &str, years: i32) -> binder::Result<String> {
Ok(format!(
"Happy Birthday {name}, congratulations with the {years} years!"
))
}
}
birthday_service/Android.bp:
rust_library {
name: "libbirthdayservice",
srcs: ["src/lib.rs"],
crate_name: "birthdayservice",
rustlibs: [
"com.example.birthdayservice-rust",
"libbinder_rs",
],
}
AIDL ์๋ฒ
๋ง์ง๋ง์ผ๋ก ์๋น์ค๋ฅผ ์ ๊ณตํ๋ ์๋ฒ๋ฅผ ๋ง๋ค ์ ์์ต๋๋ค:
birthday_service/src/server.rs:
//! Birthday service.
use birthdayservice::BirthdayService;
use com_example_birthdayservice::aidl::com::example::birthdayservice::IBirthdayService::BnBirthdayService;
use com_example_birthdayservice::binder;
const SERVICE_IDENTIFIER: &str = "birthdayservice";
/// Entry point for birthday service.
fn main() {
let birthday_service = BirthdayService;
let birthday_service_binder = BnBirthdayService::new_binder(
birthday_service,
binder::BinderFeatures::default(),
);
binder::add_service(SERVICE_IDENTIFIER, birthday_service_binder.as_binder())
.expect("Failed to register service");
binder::ProcessState::join_thread_pool()
}
birthday_service/Android.bp:
rust_binary {
name: "birthday_server",
crate_name: "birthday_server",
srcs: ["src/server.rs"],
rustlibs: [
"com.example.birthdayservice-rust",
"libbinder_rs",
"libbirthdayservice",
],
prefer_rlib: true,
}
๋ฐฐํฌ
์๋น์ค๋ฅผ ๋น๋ํ๊ณ , ๊ฐ์ ๋๋ฐ์ด์ค์ ๋ฃ๊ณ , ์์ ํ ์ ์์ต๋๋ค:
m birthday_server
adb push "$ANDROID_PRODUCT_OUT/system/bin/birthday_server /data/local/tmp"
adb shell /data/local/tmp/birthday_server
๋ค๋ฅธ ํฐ๋ฏธ๋์ ๋์์ ์๋น์ค๊ฐ ์ ์ํ๋๊ณ ์๋์ง ํ์ธํฉ๋๋ค:
adb shell service check birthdayservice
Service birthdayservice: found
service call
๋ช
๋ ์ด๋ก ์๋น์ค๋ฅผ ํธ์ถํ ์๋ ์์ต๋๋ค:
adb shell service call birthdayservice 1 s16 Bob i32 24
Result: Parcel(
0x00000000: 00000000 00000036 00610048 00700070 '....6...H.a.p.p.'
0x00000010: 00200079 00690042 00740072 00640068 'y. .B.i.r.t.h.d.'
0x00000020: 00790061 00420020 0062006f 0020002c 'a.y. .B.o.b.,. .'
0x00000030: 006f0063 0067006e 00610072 00750074 'c.o.n.g.r.a.t.u.'
0x00000040: 0061006c 00690074 006e006f 00200073 'l.a.t.i.o.n.s. .'
0x00000050: 00690077 00680074 00740020 00650068 'w.i.t.h. .t.h.e.'
0x00000060: 00320020 00200034 00650079 00720061 ' .2.4. .y.e.a.r.'
0x00000070: 00210073 00000000 's.!..... ')
AIDL ํด๋ผ์ด์ธํธ
๋ง์ง๋ง์ผ๋ก, ์๊น ์ถ๊ฐํ ์๋น์ค์ ๋ํ ํด๋ผ์ด์ธํธ๋ฅผ ๋ฌ์คํธ๋ก ๋ง๋ค๊ฒ ์ต๋๋ค.
birthday_service/src/client.rs:
//! Birthday service.
use com_example_birthdayservice::aidl::com::example::birthdayservice::IBirthdayService::IBirthdayService;
use com_example_birthdayservice::binder;
const SERVICE_IDENTIFIER: &str = "birthdayservice";
/// Connect to the BirthdayService.
pub fn connect() -> Result<binder::Strong<dyn IBirthdayService>, binder::StatusCode> {
binder::get_interface(SERVICE_IDENTIFIER)
}
/// Call the birthday service.
fn main() -> Result<(), binder::Status> {
let name = std::env::args()
.nth(1)
.unwrap_or_else(|| String::from("Bob"));
let years = std::env::args()
.nth(2)
.and_then(|arg| arg.parse::<i32>().ok())
.unwrap_or(42);
binder::ProcessState::start_thread_pool();
let service = connect().expect("Failed to connect to BirthdayService");
let msg = service.wishHappyBirthday(&name, years)?;
println!("{msg}");
Ok(())
}
birthday_service/Android.bp:
rust_binary {
name: "birthday_client",
crate_name: "birthday_client",
srcs: ["src/client.rs"],
rustlibs: [
"com.example.birthdayservice-rust",
"libbinder_rs",
],
prefer_rlib: true,
}
ํด๋ผ์ด์ธํธ๋ libbirthdayservice
์ ์์กดํ์ง ์์์ ์ฃผ๋ชฉํ์ธ์.
๋น๋ํ๊ณ , ๊ฐ์ ๋๋ฐ์ด์ค๋ก ๋ฃ๊ณ , ์คํํฉ๋๋ค:
m birthday_client
adb push "$ANDROID_PRODUCT_OUT/system/bin/birthday_client /data/local/tmp"
adb shell /data/local/tmp/birthday_client Charlie 60
Happy Birthday Charlie, congratulations with the 60 years!
API ์์
API๋ฅผ ํ์ฅํ์ฌ ๋ ๋ง์ ๊ธฐ๋ฅ์ ์ ๊ณตํด ๋ด ์๋ค. ํด๋ผ์ด์ธํธ๊ฐ ์์ผ ์นด๋์ ๋ด๊ธธ ๋ด์ฉ์ ์ง์ ํ ์ ์๋๋ก ํ๊ฒ ์ต๋๋ค:
package com.example.birthdayservice;
/** Birthday service interface. */
interface IBirthdayService {
/** Generate a Happy Birthday message. */
String wishHappyBirthday(String name, int years, in String[] text);
}
๋ก๊น
log
ํฌ๋ ์ดํธ๋ฅผ ์ฌ์ฉํ๋ฉด ์๋๋ก์ด๋ ๋๋ฐ์ด์ค ์์์ ์ํ๋ ๋์๋ logcat
์ผ๋ก, ํธ์คํธ์์ ์ํ๋ ๋์๋ stdout
์ผ๋ก ๋ก๊ทธ๊ฐ ์๋์ผ๋ก ์ถ๋ ฅ์ด ๋๋๋ก ํ ์ ์์ต๋๋ค:
hello_rust_logs/Android.bp:
rust_binary {
name: "hello_rust_logs",
crate_name: "hello_rust_logs",
srcs: ["src/main.rs"],
rustlibs: [
"liblog_rust",
"liblogger",
],
prefer_rlib: true,
host_supported: true,
}
hello_rust_logs/src/main.rs:
//! Rust logging demo.
use log::{debug, error, info};
/// Logs a greeting.
fn main() {
logger::init(
logger::Config::default()
.with_tag_on_device("rust")
.with_min_level(log::Level::Trace),
);
debug!("Starting program.");
info!("Things are going fine.");
error!("Something went wrong!");
}
๋น๋ํ๊ณ , ๊ฐ์ ๋๋ฐ์ด์ค์ ๋ฃ๊ณ , ์คํํฉ๋๋ค:
m hello_rust_logs
adb push "$ANDROID_PRODUCT_OUT/system/bin/hello_rust_logs /data/local/tmp"
adb shell /data/local/tmp/hello_rust_logs
adb logcat
์ปค๋งจ๋๋ก ๋ก๊ทธ๋ฅผ ํ์ธํฉ๋๋ค:
adb logcat -s rust
09-08 08:38:32.454 2420 2420 D rust: hello_rust_logs: Starting program.
09-08 08:38:32.454 2420 2420 I rust: hello_rust_logs: Things are going fine.
09-08 08:38:32.454 2420 2420 E rust: hello_rust_logs: Something went wrong!
์ํธ์ด์ฉ์ฑ
๋ฌ์คํธ๋ ๋ค์๊ณผ ๊ฐ์ด ๋ค๋ฅธ ์ธ์ด์์ ์ํธ์ด์ฉ์ฑ์ ํ๋ฅญํ ์ง์ํฉ๋๋ค:
- ํ ์ธ์ด์์ ๋ฌ์คํธ ํจ์๋ฅผ ํธ์ถํฉ๋๋ค.
- ํ ์ธ์ด์ ํจ์๋ฅผ ๋ฌ์คํธ์์ ํธ์ถํฉ๋๋ค.
ํ ์ธ์ด์ ํจ์๋ฅผ ํธ์ถํด์ ์ฌ์ฉํ๋ ๊ฒ์ FFI(foreign function interface)๋ผ๊ณ ํฉ๋๋ค.
C์์ ์ํธ์ด์ฉ์ฑ
๋ฌ์คํธ๋ C ํธ์ถ๊ท์ฝ์ ๋ฐ๋ฅด๋ ์ค๋ธ์ ํธ ํ์ผ๊ณผ ๋งํนํ ์ ์์ต๋๋ค. ๋ํ, ๋ฐ๋๋ก ๋ฌ์คํธ ํจ์๋ฅผ ๋ด๋ณด๋ด์ C์์ ํธ์ถ ํ ์ ๋ ์์ต๋๋ค.
์ํ๋ค๋ฉด ์๋์ ๊ฐ์ด ์๋์ผ๋ก ์ฝ๋ฉํ ์ ์์ต๋๋ค:
extern "C" { fn abs(x: i32) -> i32; } fn main() { let x = -42; let abs_x = unsafe { abs(x) }; println!("{x}, {abs_x}"); }
์ฐ๋ฆฌ๋ ์ด๋ฏธ Safe FFI ๋ํผ ์ฐ์ต๋ฌธ์ ์์ ์ด๋ฅผ ๋ค๋ฃจ์์ต๋๋ค.
์ด๋ฌํ ๋ฐฉ๋ฒ์ ํ๊ฒ ํ๋ซํผ์ ๋ชจ๋ ๋ถ๋ถ์ ์ฌ์ ์ ์๊ณ ์๋ค๋ ์ ์ ๋ฅผ ๊น๊ณ ์์ต๋๋ค. ์์ฉ ํ๋ก์ ํธ์์๋ ๊ถ์ฅํ์ง ์์ต๋๋ค.
์ข ๋ ๋์ ์ต์ ์ ์ดํด๋ณด๊ฒ ์ต๋๋ค.
Bindgen ์ฌ์ฉํ๊ธฐ
bindgen๋ C ํค๋ํ์ผ์์ ๋ฌ์คํธ ๋ฐ์ธ๋ฉ์ ์๋์ผ๋ก ์์ฑํ๋ ๋๊ตฌ์ ๋๋ค.
๋จผ์ ์์ C๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ๋ง๋ค์ด ๋ณด๊ฒ ์ต๋๋ค:
interoperability/bindgen/libbirthday.h:
typedef struct card {
const char* name;
int years;
} card;
void print_card(const card* card);
interoperability/bindgen/libbirthday.c:
#include <stdio.h>
#include "libbirthday.h"
void print_card(const card* card) {
printf("+--------------\n");
printf("| Happy Birthday %s!\n", card->name);
printf("| Congratulations with the %i years!\n", card->years);
printf("+--------------\n");
}
Android.bp
ํ์ผ์ ์๋๋ฅผ ์ถ๊ฐํฉ๋๋ค:
interoperability/bindgen/Android.bp:
cc_library {
name: "libbirthday",
srcs: ["libbirthday.c"],
}
๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ๋ํ ํค๋ ํ์ผ์ ๋ง๋ญ๋๋ค(์ด ์์์์๋ ๋ฐ๋์ ํ์ํ ๊ฒ์ ์๋๋๋ค.):
interoperability/bindgen/libbirthday_wrapper.h:
#include "libbirthday.h"
์ด์ ๋ฐ์ธ๋ฉ์ ์๋์ผ๋ก ์์ฑํ ์ ์์ต๋๋ค:
interoperability/bindgen/Android.bp:
rust_bindgen {
name: "libbirthday_bindgen",
crate_name: "birthday_bindgen",
wrapper_src: "libbirthday_wrapper.h",
source_stem: "bindings",
static_libs: ["libbirthday"],
}
๋ง์นจ๋ด, ๋ฌ์คํธ ํ๋ก๊ทธ๋จ์์ ๋ฐ์ธ๋ฉ์ ์ฌ์ฉํ ์ ์์ต๋๋ค:
interoperability/bindgen/Android.bp:
rust_binary {
name: "print_birthday_card",
srcs: ["main.rs"],
rustlibs: ["libbirthday_bindgen"],
}
interoperability/bindgen/main.rs:
//! Bindgen demo. use birthday_bindgen::{card, print_card}; fn main() { let name = std::ffi::CString::new("Peter").unwrap(); let card = card { name: name.as_ptr(), years: 42, }; unsafe { print_card(&card as *const card); } }
๋น๋ํ๊ณ , ๊ฐ์ ๋๋ฐ์ด์ค์ ๋ฃ๊ณ , ์คํํฉ๋๋ค:
m print_birthday_card
adb push "$ANDROID_PRODUCT_OUT/system/bin/print_birthday_card /data/local/tmp"
adb shell /data/local/tmp/print_birthday_card
๋ง์ง๋ง์ผ๋ก, ๋ฐ์ธ๋ฉ์ด ์ ์๋ํ๋์ง ํ์ธํ๊ธฐ ์ํด, ์๋ ์์ฑ๋ ํ ์คํธ๋ฅผ ์คํํด ๋ณด๊ฒ ์ต๋๋ค:
interoperability/bindgen/Android.bp:
rust_test {
name: "libbirthday_bindgen_test",
srcs: [":libbirthday_bindgen"],
crate_name: "libbirthday_bindgen_test",
test_suites: ["general-tests"],
auto_gen_config: true,
clippy_lints: "none", // Generated file, skip linting
lints: "none",
}
atest libbirthday_bindgen_test
C์์ ๋ฌ์คํธ ํธ์ถ
๋ฌ์คํธ์์ ํ์ ๊ณผ ํจ์๋ฅผ C๋ก ๋ด๋ณด๋ด๋ ๊ฒ์ ๊ฐ๋จํฉ๋๋ค:
interoperability/rust/libanalyze/analyze.rs
//! Rust FFI demo. #![deny(improper_ctypes_definitions)] use std::os::raw::c_int; /// Analyze the numbers. #[no_mangle] pub extern "C" fn analyze_numbers(x: c_int, y: c_int) { if x < y { println!("x ({x}) is smallest!"); } else { println!("y ({y}) is probably larger than x ({x})"); } }
interoperability/rust/libanalyze/analyze.h
#ifndef ANALYSE_H
#define ANALYSE_H
extern "C" {
void analyze_numbers(int x, int y);
}
#endif
interoperability/rust/libanalyze/Android.bp
rust_ffi {
name: "libanalyze_ffi",
crate_name: "analyze_ffi",
srcs: ["analyze.rs"],
include_dirs: ["."],
}
์ด์ ์ด ๋ฌ์คํธ ํจ์๋ฅผ C๋ฐ์ด๋๋ฆฌ์์ ํธ์ถํ ์ ์์ต๋๋ค:
interoperability/rust/analyze/main.c
#include "analyze.h"
int main() {
analyze_numbers(10, 20);
analyze_numbers(123, 123);
return 0;
}
interoperability/rust/analyze/Android.bp
cc_binary {
name: "analyze_numbers",
srcs: ["main.c"],
static_libs: ["libanalyze_ffi"],
}
๋น๋ํ๊ณ , ๊ฐ์ ๋๋ฐ์ด์ค์ ๋ฃ๊ณ , ์คํํฉ๋๋ค:
m analyze_numbers
adb push "$ANDROID_PRODUCT_OUT/system/bin/analyze_numbers /data/local/tmp"
adb shell /data/local/tmp/analyze_numbers
#[no_mangle]
์ ๋ฌ์คํธ์ ๋ค์ ๋งน๊ธ๋ง(name mangling)์ ๋นํ์ฑํํ๋ฏ๋ก ์ธ๋ถ๋ก ๋
ธ์ถ๋๋ ์ฌ๋ณผ์ ์ด๋ฆ์ ํจ์์ ์ด๋ฆ ๊ทธ๋๋ก๊ฐ ๋ฉ๋๋ค. ์ฌ๋ณผ ์ด๋ฆ์ ๋ฐ๊พธ๊ณ ์ถ๋ค๋ฉด #[export_name = "some_name"]
์ ์ฌ์ฉํฉ๋๋ค.
C++์์ ์ํธ์ด์ฉ์ฑ
CXX ํฌ๋ ์ดํธ๋ ๋ฌ์คํธ์ C++ ์ฌ์ด์ ์์ ํ ์ํธ์ด์ฉ์ฑ์ ๊ฐ๋ฅํ๊ฒ ํด์ค๋๋ค.
์ ์ฒด์ ์ธ ์ ๊ทผ ๋ฐฉ๋ฒ์ ๋ค์๊ณผ ๊ฐ์ต๋๋ค:
์ฌ์ฉํ๋ ๋ฐฉ๋ฒ์ ๋ํด์๋CXX ํํ ๋ฆฌ์ผ ๋ฅผ ์ฐธ์กฐํฉ๋๋ค.
-
At this point, the instructor should switch to the CXX tutorial.
-
Walk the students through the tutorial step by step.
-
Highlight how CXX presents a clean interface without unsafe code in both languages.
-
Show the correspondence between Rust and C++ types:
-
Explain how a Rust
String
cannot map to a C++std::string
(the latter does not uphold the UTF-8 invariant). Show that despite being different types,rust::String
in C++ can be easily constructed from a C++std::string
, making it very ergonomic to use. -
Explain that a Rust function returning
Result<T, E>
becomes a function which throws aE
exception in C++ (and vice versa).
-
Java์์ ์ํธ์ด์ฉ์ฑ
์๋ฐ๋ Java Native Interface(JNI)๋ฅผ ํตํด ๊ณต์ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ๋ก๋ํ ์ ์์ต๋๋ค. jni
ํฌ๋ ์ดํธ๋ฅผ ์ฌ์ฉํ์ฌ JNI ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ๋ง๋ค ์ ์์ต๋๋ค.
๋จผ์ , ์๋ฐ๋ก ๋ด๋ณด๋ผ ๋ฌ์คํธ ํจ์๋ฅผ ์์ฑํฉ๋๋ค:
interoperability/java/src/lib.rs:
#![allow(unused)] fn main() { //! Rust <-> Java FFI demo. use jni::objects::{JClass, JString}; use jni::sys::jstring; use jni::JNIEnv; /// HelloWorld::hello method implementation. #[no_mangle] pub extern "system" fn Java_HelloWorld_hello( env: JNIEnv, _class: JClass, name: JString, ) -> jstring { let input: String = env.get_string(name).unwrap().into(); let greeting = format!("Hello, {input}!"); let output = env.new_string(greeting).unwrap(); output.into_inner() } }
interoperability/java/Android.bp:
rust_ffi_shared {
name: "libhello_jni",
crate_name: "hello_jni",
srcs: ["src/lib.rs"],
rustlibs: ["libjni"],
}
์๋ฐ์์ ์ด ํจ์๋ฅผ ํธ์ถ ํ ์ ์์ต๋๋ค:
interoperability/java/HelloWorld.java:
class HelloWorld {
private static native String hello(String name);
static {
System.loadLibrary("hello_jni");
}
public static void main(String[] args) {
String output = HelloWorld.hello("Alice");
System.out.println(output);
}
}
interoperability/java/Android.bp:
java_binary {
name: "helloworld_jni",
srcs: ["HelloWorld.java"],
main_class: "HelloWorld",
required: ["libhello_jni"],
}
๋ง์ง๋ง์ผ๋ก ๋ฐ์ด๋๋ฆฌ๋ฅผ ๋น๋, ์ฑํฌ, ์คํํฉ๋๋ค:
m helloworld_jni
adb sync # requires adb root && adb remount
adb shell /system/bin/helloworld_jni
์ฐ์ต๋ฌธ์
This is a group exercise: We will look at one of the projects you work with and try to integrate some Rust into it. Some suggestions:
-
๋น์ ์ AIDL์๋น์ค๋ฅผ ๋ฌ์คํธ ํด๋ผ์ด์ธํธ์์ ํธ์ถํด๋ด ๋๋ค.
-
๋น์ ์ ํ๋ก์ ํธ์ ํจ์๋ฅผ ๋ฌ์คํธ๋ก ์ฎ๊ธฐ๊ณ ํธ์ถํด๋ด ๋๋ค.
์ด ์ฐ์ต๋ฌธ์ ๋ ์ด๋ ค์๊ธฐ ๋๋ฌธ์ ํด๋ต์ด ์ ๊ณต๋์ง ์์ต๋๋ค. ํด๋์ค์์ ์ ์ถ๋ ์ฝ๋์ ์์กดํฉ๋๋ค.
Welcome to Bare Metal Rust
์ด ๊ณผ์ ์ Rust์ ๋ํด ์ด๋์ ๋ ๊ฒฝํ์ด ์๊ณ (์๋ง๋ Comprehensive Rust ๊ณผ์ ์ ํตํด) C์ ๊ฐ์ ๋ค๋ฅธ ์ธ์ด๋ก bare-metal ํ๋ก๊ทธ๋๋ฐ์ ํด ๋ณธ ์ฌ์ฉ์๋ฅผ ๋์์ผ๋ก ํ๋ bare-metal Rust์ ๊ดํ ๋ ๋ฆฝ์ ์ธ 1์ผ ๊ณผ์ ์ ๋๋ค.
์ค๋์ OS๋ฅผ ์ฌ์ฉํ์ง ์๊ณ Rust ์ฝ๋๋ฅผ ์คํํ๋ โbare-metalโ Rust์ ๊ดํด ์์๋ด ๋๋ค. ๋ณธ ๊ฐ์์ ๊ตฌ์ฑ์ ๋ค์๊ณผ ๊ฐ์ต๋๋ค.
no_std
Rust๋ ๋ฌด์์ธ๊ฐ์?- ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์ฉ ํ์จ์ด ์์ฑ
- ์ ํ๋ฆฌ์ผ์ด์ ํ๋ก์ธ์๋ฅผ ์ํ ๋ถํธ๋ก๋ / ์ปค๋ ์ฝ๋ ์์ฑ
- bare-metal Rust ๊ฐ๋ฐ์ ์ํ ์ ์ฉํ ํฌ๋ ์ดํธ
์ด ๊ฐ์์์๋ BBC micro:bit v2๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ๋ฅผ ์ฌ์ฉํฉ๋๋ค. ์ด ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ๋ Nordic nRF51822 ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์ ๊ธฐ๋ฐํ ๊ฐ๋ฐ ๋ณด๋๋ก์จ, LED์ ๋ฒํผ, I2C ์ฐ๊ฒฐ ๊ฐ์๋๊ณ ๋ฐ ๋์นจ๋ฐ, ์จ๋ณด๋ SWD ๋๋ฒ๊ฑฐ๋ฅผ ํฌํจํ๊ณ ์์ต๋๋ค.
์์ํ๊ธฐ์ ์, ์์ผ๋ก ์ฌ์ฉํ ๋๊ตฌ๋ฅผ ์ค์นํด์ผ ํฉ๋๋ค. gLinux ๋๋ Debian๋ฅผ ์ฌ์ฉํ๊ณ ์๋ค๋ฉด ์๋์ ๊ฐ์ด ํ์ธ์.
sudo apt install gcc-aarch64-linux-gnu gdb-multiarch libudev-dev picocom pkg-config qemu-system-arm
rustup update
rustup target add aarch64-unknown-none thumbv7em-none-eabihf
rustup component add llvm-tools-preview
cargo install cargo-binutils cargo-embed
plugdev
๊ทธ๋ฃน์ ์ฌ์ฉ์์๊ฒ micro:bit ํ๋ก๊ทธ๋๋จธ ์ฅ์น์ ๋ํ ์ก์ธ์ค ๊ถํ์ ๋ถ์ฌํฉ๋๋ค.
echo 'SUBSYSTEM=="usb", ATTR{idVendor}=="0d28", MODE="0664", GROUP="plugdev"' |\
sudo tee /etc/udev/rules.d/50-microbit.rules
sudo udevadm control --reload-rules
MacOS์์:
xcode-select --install
brew install gdb picocom qemu
brew install --cask gcc-aarch64-embedded
rustup update
rustup target add aarch64-unknown-none thumbv7em-none-eabihf
rustup component add llvm-tools-preview
cargo install cargo-binutils cargo-embed
no_std
|
|
|
---|---|---|
|
|
|
HashMap
์ RNG์ ์์กดํฉ๋๋ค.std
๋core
๋ฐalloc
๋ฅผ ํฌํจํฉ๋๋ค.
์ต์ํ์ no_std
ํ๋ก๊ทธ๋จ
#![no_main] #![no_std] use core::panic::PanicInfo; #[panic_handler] fn panic(_panic: &PanicInfo) -> ! { loop {} }
- ์ด ์ฝ๋๋ ๋น ๋ฐ์ด๋๋ฆฌ๋ก ์ปดํ์ผ๋ฉ๋๋ค.
std
๋ ํจ๋ ํธ๋ค๋ฌ๋ฅผ ์ ๊ณตํ์ง๋ง, ์ฐ๋ฆฌ๋ ์์ฒด์ ์ผ๋ก ํธ๋ค๋ฌ๋ฅผ ๋ง๋ค์ด์ผํฉ๋๋ค.- ํจ๋ ํธ๋ค๋ฌ๋
panic-halt
์ ๊ฐ์ ํฌ๋ ์ดํธ๋ฅผ ํตํด์ ๋ง๋ค์๋ ์์ต๋๋ค. - ํ๊ฒ์ ๋ฐ๋ผ
panic = "abort"
๋ก ์ปดํ์ผํด์ผ ํ ์ ์์ต๋๋ค. ์ด๋eh_personality
์ ๊ดํ ์ค๋ฅ๋ฅผ ๋ฐฉ์งํ๊ธฐ ์ํจ์ ๋๋ค. main
๊ณผ ๊ฐ์ ํ๋ก๊ทธ๋จ ์ง์ ์ ์ด ์์ต๋๋ค. ๊ฐ๋ฐ์๊ฐ ์์ฒด์ ์ผ๋ก ์ง์ ์ ์ ์ ์ํด์ผ ํฉ๋๋ค. ์ง์ ์ ์ ์ ์ํ๋ ์์ ์, ์ผ๋ฐ์ ์ผ๋ก ๋ง์ปค ์คํฌ๋ฆฝํธ์ ์ด์ ๋ธ๋ฆฌ ์ฝ๋๋ฅผ ํ์๋ก ํฉ๋๋ค.
alloc
alloc
์ ์ฌ์ฉํ๋ ค๋ฉด ์ ์ญ (ํ) ํ ๋น์๋ฅผ ๊ตฌํํด์ผ ํฉ๋๋ค.
#![no_main] #![no_std] extern crate alloc; extern crate panic_halt as _; use alloc::string::ToString; use alloc::vec::Vec; use buddy_system_allocator::LockedHeap; #[global_allocator] static HEAP_ALLOCATOR: LockedHeap<32> = LockedHeap::<32>::new(); static mut HEAP: [u8; 65536] = [0; 65536]; pub fn entry() { // Safe because `HEAP` is only used here and `entry` is only called once. unsafe { // Give the allocator some memory to allocate. HEAP_ALLOCATOR .lock() .init(HEAP.as_mut_ptr() as usize, HEAP.len()); } // Now we can do things that require heap allocation. let mut v = Vec::new(); v.push("A string".to_string()); }
buddy_system_allocator
๋ ๊ฐ๋จํ ๋ฒ๋ ์์คํ ํ ๋น์๋ฅผ ๊ตฌํํ๋ ์๋ ํํฐ ํฌ๋ ์ดํธ์ ๋๋ค. ์ด ์ธ์๋, ๋ค๋ฅธ ํฌ๋ ์ดํธ๋ฅผ ์ฌ์ฉํ๊ฑฐ๋, ์ง์ ํ ๋น์๋ฅผ ๋ง๋ค๊ฑฐ๋, ์ด๋ฏธ ์กด์ฌํ๋ ๋ค๋ฅธ ํ ๋น์์ ํํนํ ์ ์์ต๋๋ค.LockHeap
ํ์ ์ const ๋งค๊ฐ๋ณ์๋ ํ ๋น์์ ์ต๋ ํฌ๊ธฐ๋ฅผ 2์ง์๋ก ํํํ์ ๋์ ์๋ฆฟ์์ ๋๋ค. ์ฆ, ์ด ๊ฒฝ์ฐ์ฒ๋ผ 32์ธ ๊ฒฝ์ฐ ์ต๋ 2**32๋ฐ์ดํธ ํฌ๊ธฐ์ ์์ญ์ ๋ค๋ฃฐ ์ ์์ต๋๋ค.- ํ ๋ฐ์ด๋๋ฆฌ์์
alloc
์ ์์กดํ๋ ํฌ๋ ์ดํธ๊ฐ ํ๋๋ผ๋ ์๋ค๋ฉด ๋ฐ์ด๋๋ฆฌ ์ ์ฒด์์ ์ ์ญ ํ ๋น์๊ฐ ๋ฐ๋์ ํ๋ ์กด์ฌํด์ผ ํฉ๋๋ค. ์ผ๋ฐ์ ์ผ๋ก ์ ์ญ ํ ๋น์๋ฅผ ์ ์ธํ๋ ์์ ์ ์ต์์ ๋ฐ์ด๋๋ฆฌ ํฌ๋ ์ดํธ์์ ์ด๋ฃจ์ด์ง๋๋ค. panic_halt
ํฌ๋ ์ดํธ๊ฐ ์ฐ๊ฒฐ๋์ด ํจ๋ ํธ๋ค๋ฌ๋ฅผ ๊ฐ์ ธ์ค๋๋ก ํ๋ ค๋ฉดextern crate panic_halt as _
๊ฐ ํ์ํฉ๋๋ค.- ์ด ์์ ์ฝ๋๋ ๋น๋๋ ๋์ง๋ง, ์ง์ ์ ์ด ์๊ธฐ ๋๋ฌธ์ ์คํ๋์ง๋ ์์ต๋๋ค.
๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ
โcortex_m_rtโ ํฌ๋ ์ดํธ๋ Cortex M ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ๋ฅผ ์ด๊ธฐํ ํ๋ ํธ๋ค๋ฌ๋ฅผ ์ ๊ณตํฉ๋๋ค.
#![no_main] #![no_std] extern crate panic_halt as _; mod interrupts; use cortex_m_rt::entry; #[entry] fn main() -> ! { loop {} }
์ด์ , ์ฃผ๋ณ์ฅ์น์ ์ก์ธ์คํ๋ ๋ฐฉ๋ฒ์ ์์๋ณด๊ฒ ์ต๋๋ค. ๊ฐ์ฅ ๊ธฐ๊ณ์ ๊ฐ๊น์ด ๋ฎ์ ๋จ๊ณ์์ ์์ํด์ ์ ์ ์ถ์ํ ์์ค์ ์ฌ๋ฆฌ๊ฒ ์ต๋๋ค.
cortex_m_rt::entry
๋งคํฌ๋ก๋ ์ง์ ์ ์ผ๋ก ์ฌ์ฉ๋๋ ํจ์๊ฐfn() -> !
ํ์ (์ฆ, ๋ฆฌํดํ์ง ์๋)์์ ์๊ตฌํฉ๋๋ค. ๋ง์ฝ, ๋ฆฌํดํ๊ฒ ๋๋ฉด, ํ๋ก๊ทธ๋จ ์ํ ํ ๋ฆฌ์ ํธ๋ค๋ฌ๋ก ๋์๊ฐ๊ฒ ๋๋ ๊ฒ์ธ๋ฐ ์ด๋ ๋ง์ด ๋์ง ์๊ธฐ ๋๋ฌธ์ ๋๋ค.cargo embed --bin minimal
์ ์ฌ์ฉํ์ฌ ์์๋ฅผ ์คํํฉ๋๋ค.
์์ MMIO
๋๋ถ๋ถ์ ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ๋ ๋ฉ๋ชจ๋ฆฌ ๋งคํ IO๋ฅผ ํตํด ์ฃผ๋ณ๊ธฐ๊ธฐ์ ์ก์ธ์คํฉ๋๋ค. micro:bit์์ LED๋ฅผ ์ผ๋ณด๊ฒ ์ต๋๋ค.
#![no_main] #![no_std] extern crate panic_halt as _; mod interrupts; use core::mem::size_of; use cortex_m_rt::entry; /// GPIO port 0 peripheral address const GPIO_P0: usize = 0x5000_0000; // GPIO peripheral offsets const PIN_CNF: usize = 0x700; const OUTSET: usize = 0x508; const OUTCLR: usize = 0x50c; // PIN_CNF fields const DIR_OUTPUT: u32 = 0x1; const INPUT_DISCONNECT: u32 = 0x1 << 1; const PULL_DISABLED: u32 = 0x0 << 2; const DRIVE_S0S1: u32 = 0x0 << 8; const SENSE_DISABLED: u32 = 0x0 << 16; #[entry] fn main() -> ! { // Configure GPIO 0 pins 21 and 28 as push-pull outputs. let pin_cnf_21 = (GPIO_P0 + PIN_CNF + 21 * size_of::<u32>()) as *mut u32; let pin_cnf_28 = (GPIO_P0 + PIN_CNF + 28 * size_of::<u32>()) as *mut u32; // Safe because the pointers are to valid peripheral control registers, and // no aliases exist. unsafe { pin_cnf_21.write_volatile( DIR_OUTPUT | INPUT_DISCONNECT | PULL_DISABLED | DRIVE_S0S1 | SENSE_DISABLED, ); pin_cnf_28.write_volatile( DIR_OUTPUT | INPUT_DISCONNECT | PULL_DISABLED | DRIVE_S0S1 | SENSE_DISABLED, ); } // Set pin 28 low and pin 21 high to turn the LED on. let gpio0_outset = (GPIO_P0 + OUTSET) as *mut u32; let gpio0_outclr = (GPIO_P0 + OUTCLR) as *mut u32; // Safe because the pointers are to valid peripheral control registers, and // no aliases exist. unsafe { gpio0_outclr.write_volatile(1 << 28); gpio0_outset.write_volatile(1 << 21); } loop {} }
- GPIO 0 ํ 21์ LED ๋งคํธ๋ฆญ์ค์ ์ฒซ ๋ฒ์งธ ์ด์ ์ฐ๊ฒฐ๋๊ณ ํ 28์ ์ฒซ ๋ฒ์งธ ํ์ ์ฐ๊ฒฐ๋ฉ๋๋ค.
์๋ ๋ช ๋ น์ด๋ก ์์ ์ฝ๋๋ฅผ ์คํํ์ธ์.
cargo embed --bin mmio
์ฃผ๋ณ๊ธฐ๊ธฐ ์ก์ธ์ค ํฌ๋ ์ดํธ
svd2rust
ํฌ๋ ์ดํธ๋ฅผ ์ด์ฉํ๋ฉด ๋ฉ๋ชจ๋ฆฌ ๋งคํ๋ ์ฃผ๋ณ์ฅ์น๋ฅผ ๊ธฐ์ ํ๋ CMSIS-SVD ํ์ผ๋ก๋ถํฐ Rust ๋ํผ๋ฅผ ์์ฑํ ์ ์์ต๋๋ค.
#![no_main] #![no_std] extern crate panic_halt as _; use cortex_m_rt::entry; use nrf52833_pac::Peripherals; #[entry] fn main() -> ! { let p = Peripherals::take().unwrap(); let gpio0 = p.P0; // Configure GPIO 0 pins 21 and 28 as push-pull outputs. gpio0.pin_cnf[21].write(|w| { w.dir().output(); w.input().disconnect(); w.pull().disabled(); w.drive().s0s1(); w.sense().disabled(); w }); gpio0.pin_cnf[28].write(|w| { w.dir().output(); w.input().disconnect(); w.pull().disabled(); w.drive().s0s1(); w.sense().disabled(); w }); // Set pin 28 low and pin 21 high to turn the LED on. gpio0.outclr.write(|w| w.pin28().clear()); gpio0.outset.write(|w| w.pin21().set()); loop {} }
- SVD(System View Description) ํ์ผ์ ์ผ๋ฐ์ ์ผ๋ก ์ค๋ฆฌ์ฝ ๊ณต๊ธ์
์ฒด์์ ์ ๊ณตํ๋ XML ํ์ผ๋ก, ๊ธฐ๊ธฐ์ ๋ฉ๋ชจ๋ฆฌ ๋งต์ ๊ธฐ์ ํฉ๋๋ค.
- ์ฃผ๋ณ๊ธฐ๊ธฐ, ๋ ์ง์คํฐ, ํ๋, ๊ฐ์ผ๋ก ๊ตฌ์ฑ๋๋ฉฐ ์ด๋ฆ, ์ค๋ช , ์ฃผ์ ๋ฑ์ด ํฌํจ๋ฉ๋๋ค.
- SVD ํ์ผ์๋ ๋ฒ๊ทธ๊ฐ ์์ ์ ์๊ณ ๋ถ์์ ํ๊ธฐ ๋๋ฌธ์, ์ด๋ฌํ ๋ฌธ์ ๋ค์ ํจ์นํ๋ ๋ค์ํ ํ๋ก์ ํธ๋ค์ด ์์ต๋๋ค.
cortex-m-rt
๋ ๋ฌด์๋ณด๋ค๋ ๋ฒกํฐ ํ ์ด๋ธ์ ์ ๊ณตํฉ๋๋ค.cargo install cargo-binutils
์ ์คํํ ํ,cargo objdump --bin pac -- -d --no-show-raw-insn
์ ์คํํ์ฌ ์์ฑ๋ ๋ฐ์ด๋๋ฆฌ์ ๋ด์ฉ์ ํ์ธํ ์ ์์ต๋๋ค.
์๋ ๋ช ๋ น์ด๋ก ์์ ์ฝ๋๋ฅผ ์คํํ์ธ์.
cargo embed --bin pac
HAL ํฌ๋ ์ดํธ๋ค
๋ค์ํ ์ฃผ๋ณ ์ฅ์น์ ๋ํ ๋ํผ๋ฅผ ์ ๊ณตํ๋ HAL ํฌ๋ ์ดํธ๋ค์ด ์์ต๋๋ค. ์ด ํฌ๋ ์ดํธ๋ค์ ์ผ๋ฐ์ ์ผ๋ก embedded-hal
์ ํธ๋ ์์ ๊ตฌํํฉ๋๋ค.
#![no_main] #![no_std] extern crate panic_halt as _; use cortex_m_rt::entry; use nrf52833_hal::gpio::{p0, Level}; use nrf52833_hal::pac::Peripherals; use nrf52833_hal::prelude::*; #[entry] fn main() -> ! { let p = Peripherals::take().unwrap(); // Create HAL wrapper for GPIO port 0. let gpio0 = p0::Parts::new(p.P0); // Configure GPIO 0 pins 21 and 28 as push-pull outputs. let mut col1 = gpio0.p0_28.into_push_pull_output(Level::High); let mut row1 = gpio0.p0_21.into_push_pull_output(Level::Low); // Set pin 28 low and pin 21 high to turn the LED on. col1.set_low().unwrap(); row1.set_high().unwrap(); loop {} }
set_low
๋ฐset_high
๋embedded_hal
OutputPin
ํธ๋ ์์ ๋ฉ์๋์ ๋๋ค.- ๋ค์ํ STM32, GD32, nRF, NXP, MSP430, AVR, PIC ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ๋ฅผ ๋น๋กฏํ ๋ง์ Cortex-M ๋ฐ RISC-V ๊ธฐ๊ธฐ๋ฅผ ์ํ HAL ํฌ๋ ์ดํธ๊ฐ ์์ต๋๋ค
์๋ ๋ช ๋ น์ด๋ก ์์ ์ฝ๋๋ฅผ ์คํํ์ธ์.
cargo embed --bin hal
Board support crates
๋ณด๋ ์ง์ ํฌ๋ ์ดํธ๋ค์, ํน์ ๋ณด๋๋ฅผ ๋ ์์ฝ๊ฒ ์ฌ์ฉํ ์ ์๊ฒ ํด ์ฃผ๋ ๋ ๋์ ์์ค์ ์ถ์ํ๋ฅผ ์ ๊ณตํฉ๋๋ค.
#![no_main] #![no_std] extern crate panic_halt as _; use cortex_m_rt::entry; use microbit::hal::prelude::*; use microbit::Board; #[entry] fn main() -> ! { let mut board = Board::take().unwrap(); board.display_pins.col1.set_low().unwrap(); board.display_pins.row1.set_high().unwrap(); loop {} }
- ์ด ๊ฒฝ์ฐ ๋ณด๋ ์ง์ ํฌ๋ ์ดํธ๋ ์ข ๋ ์ง๊ด์ ์ธ ์ด๋ฆ๋ค๊ณผ ์ ๋นํ ์์ค์ ์ด๊ธฐํ๋ฅผ ์ ๊ณตํฉ๋๋ค.
- ์ด ํฌ๋ ์ดํธ๋ ๋ง์ดํฌ๋ก์ปจํธ๋กค ๋ฐ์ ์๋ (์ฆ, ๋ณด๋์ ์ค์น๋) ์ฅ์น์ ๋ํ ๋๋ผ์ด๋ฒ๋ ํฌํจํ ์ ์์ต๋๋ค.
microbit-v2
์๋ LED ๋งคํธ๋ฆญ์ค๋ฅผ ์ํ ๊ฐ๋จํ ๋๋ผ์ด๋ฒ๊ฐ ํฌํจ๋์ด ์์ต๋๋ค.
์๋ ๋ช ๋ น์ด๋ก ์์ ์ฝ๋๋ฅผ ์คํํ์ธ์.
cargo embed --bin board_support
ํ์ ์ํ ํจํด
#[entry] fn main() -> ! { let p = Peripherals::take().unwrap(); let gpio0 = p0::Parts::new(p.P0); let pin: P0_01<Disconnected> = gpio0.p0_01; // let gpio0_01_again = gpio0.p0_01; // Error, moved. let pin_input: P0_01<Input<Floating>> = pin.into_floating_input(); if pin_input.is_high().unwrap() { // ... } let mut pin_output: P0_01<Output<OpenDrain>> = pin_input .into_open_drain_output(OpenDrainConfig::Disconnect0Standard1, Level::Low); pin_output.set_high().unwrap(); // pin_input.is_high(); // Error, moved. let _pin2: P0_02<Output<OpenDrain>> = gpio0 .p0_02 .into_open_drain_output(OpenDrainConfig::Disconnect0Standard1, Level::Low); let _pin3: P0_03<Output<PushPull>> = gpio0.p0_03.into_push_pull_output(Level::Low); loop {} }
- ํ์
Copy
๋๋Clone
์ ๊ตฌํํ์ง ์์ผ๋ฏ๋ก ๊ฐ๊ฐ ํ๋์ ์ธ์คํด์ค๋ง ์กด์ฌํ ์ ์์ต๋๋ค. ํ์ ํฌํธ ๊ตฌ์กฐ์ฒด ๋ฐ์ผ๋ก ์ด๋ํ๋ฉด ์๋ฌด๋ ์ฌ์ฉํ ์ ์์ต๋๋ค. - ํ ๊ตฌ์ฑ์ ๋ณ๊ฒฝํ๋ฉด ์ด์ ํ ์ธ์คํด์ค๊ฐ ์ฌ์ฉ๋๋ฏ๋ก ์ดํ์ ์ด์ ์ธ์คํด์ค๋ฅผ ๊ณ์ ์ฌ์ฉํ ์ ์์ต๋๋ค.
- ๊ฐ ์ ํ์ ํ์ฌ ์ํ๋ฅผ ๋ํ๋ ๋๋ค. ์๋ฅผ ๋ค์ด ์ด ๊ฒฝ์ฐ์๋ GPIO ํ์ ๊ตฌ์ฑ ์ํ์ ๋๋ค. ์ด๋ ๊ฒ ํ๋ฉด ์ํ ๋จธ์ ์ด ์ ํ ์์คํ ์ผ๋ก ์ธ์ฝ๋ฉ๋๋ฉฐ, ๋จผ์ ์ฌ๋ฐ๋ฅด๊ฒ ๊ตฌ์ฑํ ํ ํน์ ๋ฐฉ์์ผ๋ก ํ์ ์ฌ์ฉํ๋๋ก ํฉ๋๋ค. ์๋ชป๋ ์ํ ์ ํ์ ์ปดํ์ผ ์๊ฐ์ ํฌ์ฐฉ๋ฉ๋๋ค.
- ์
๋ ฅ ํ์์
is_high
๋ฅผ ํธ์ถํ๊ณ ์ถ๋ ฅ ํ์์set_high
๋ฅผ ํธ์ถํ ์ ์์ง๋ง ๊ทธ ๋ฐ๋๋ก๋ ์ ๋ฉ๋๋ค. - ๋ง์ HAL ํฌ๋ ์ดํธ๋ค์ด ์ด ํจํด์ ๋ฐ๋ฆ ๋๋ค.
embedded-hal
embedded-hal
ํฌ๋ ์ดํธ๋ ๋ค์ํ ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์์ ๊ณตํต์ ์ผ๋ก ์ฐพ์๋ณผ ์ ์๋ ์ฃผ๋ณ๊ธฐ๊ธฐ๋ฅผ ์ถ์ํ ํ๋ ๋ค์ํ ํธ๋ ์์ ์ ๊ณตํฉ๋๋ค.
- GPIO
- ADC
- I2C, SPI, UART, CAN
- RNG
- ํ์ด๋จธ
- ์์น๋
๊ทธ๋ฌ๋ฉด ๋ค๋ฅธ ํฌ๋ ์ดํธ๋ ์ด ํธ๋ ์๋ค์ ํ์ฉํ์ฌ ๋๋ผ์ด๋ฒ๋ฅผ ๊ตฌํํฉ๋๋ค. ์๋ฅผ ๋ค์ด ๊ฐ์๋๊ณ ๋๋ผ์ด๋ฒ๋ฅผ ๊ตฌํํ ๋ I2C ๋๋ SPI ๋ฒ์ค ๊ตฌํ์ ์ฌ์ฉํ ์ ์์ต๋๋ค.
- ๋ผ์ค๋ฒ ๋ฆฌ ํ์ด์์ ๋์๊ฐ๋ ๋ฆฌ๋ ์ค ๊ฐ์ ํ๋ซํผ ๋ฟ๋ง ์๋๋ผ ๋ค๋ฅธ ์ฌ๋ฌ ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์ ๋ํ ๊ตฌํ์ด ์์ต๋๋ค.
embedded-hal
์ โasyncโ ๋ฒ์ ์ ๊ดํ ์์ ์ด ์งํ ์ค์ด์ง๋ง ์์ง ์์ ์ ์ด์ง ์์ต๋๋ค.
probe-rs
, cargo-embed
probe-rs๋ ์๋ฒ ๋๋ ์์คํ ์ ์ํ ๋๊ตฌ ๋ชจ์์ ๋๋ค. OpenOCD์ ๋น์ทํ์ง๋ง, Rust์ ๋ ์ ํตํฉ๋์ด ์์ต๋๋ค.
- SWD๋ฐ JTAG(CMSIS-DAP, ST-Link, J-Link ํ๋ก๋ธ๋ฅผ ํตํจ)
- GDB ์คํ ๋ฐ Microsoft DAP์๋ฒ
- Cargo์ ํตํฉ๋จ
cargo-embed
๋ cargo์ ์๋ธ ์ปค๋งจํธ๋ก์จ, ๋ฐ์ด๋๋ฆฌ๋ฅผ ๋น๋ํ๊ณ ํ๋์ํ๋ฉฐ, RTT์ถ๋ ฅ์ ๊ธฐ๋กํ๊ณ , GDB๋ฅผ ์ฐ๊ฒฐํด ์ค๋๋ค. ์ด ๋ช
๋ น์ด์ ์ธ๋ถ ๋์์ ํ๋ก์ ํธ ๋๋ ํฐ๋ฆฌ์ Embed.toml
ํ์ผ์ ํตํด ์ค์ ํฉ๋๋ค.
- CMSIS-DAP๋ Arm์์ ์ ์ํ ํ๋กํ ์ฝ๋ก, USB๋ฅผ ํตํด Arm Cortex ํ๋ก์ธ์์ CoreSight ๋๋ฒ๊ทธ ์ก์ธ์ค ํฌํธ์ ์ ๊ทผํ ์ ์๊ฒ ํด ์ค๋๋ค. BBC micro:bit์ ์๋ ์จ๋ณด๋ ๋๋ฒ๊ฑฐ๋ ์ด ํ๋กํ ์ฝ์ ์ง์ํฉ๋๋ค.
- ST-Link๋ ST Microelectronics์ฌ์์ ๋ง๋ in-circuit ๋๋ฒ๊ฑฐ๋ค์ด๋ฉฐ, J-Link๋ SEGGER์ฌ์ in-circuit ๋๋ฒ๊ฑฐ๋ค์ ๋๋ค.
- ๋๋ฒ๊ทธ ์ก์ธ์ค ํฌํธ์ ๋ฌผ๋ฆฌ์ ์ธ ๊ตฌ์ฑ์ ์ผ๋ฐ์ ์ผ๋ก 5ํ JTAG ์ธํฐํ์ด์ค ํน์, 2ํ Serial Wire Debug ์ธํฐํ์ด์ค ์ ๋๋ค.
- probe-rs๋ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ก ๊ตฌํ๋์ด ์์ด์, ๋ค๋ฅธ ๋๊ตฌ๋ค์ ํตํฉ๋๊ธฐ๊ฐ ์ฝ์ต๋๋ค.
- Microsoft ๋๋ฒ๊ทธ ์ด๋ํฐ ํ๋กํ ์ฝ์ ์ฌ์ฉํ๋ฉด VSCode๋ ๋ค๋ฅธ IDE ์์์ ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์์ ์ํ์ค์ธ ์ฝ๋๋ฅผ ๋๋ฒ๊น ํ ์ ์์ต๋๋ค.
- cargo-embed๋ probe-rs ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ฌ์ฉํ์ฌ ๋น๋๋ ๋ฐ์ด๋๋ฆฌ์ ๋๋ค.
- RTT(Real Time Transfers)๋ ์ฌ๋ฌ ๋ง ๋ฒํผ๋ฅผ ํตํด ๋๋ฒ๊ทธ ํธ์คํธ์ ํ๊ฒ ๊ฐ์ ๋ฐ์ดํฐ๋ฅผ ์ ์กํ๋ ๋ฉ์ปค๋์ฆ์ ๋๋ค.
๋๋ฒ๊น
Embed.toml:
[default.general]
chip = "nrf52833_xxAA"
[debug.gdb]
enabled = true
src/bare-metal/microcontrollers/examples/
์๋ ํ๋์ ํฐ๋ฏธ๋์์:
cargo embed --bin board_support debug
In another terminal in the same directory:
gdb-multiarch target/thumbv7em-none-eabihf/debug/board_support --eval-command="target remote :1337"
GDB์์ ๋ค์์ ์คํํด ๋ณด์ธ์.
b src/bin/board_support.rs:29
b src/bin/board_support.rs:30
b src/bin/board_support.rs:32
c
c
c
๋ค๋ฅธ ํ๋ก์ ํธ
- RTIC
- โ์ค์๊ฐ ์ธํฐ๋ฝํธ ๊ธฐ๋ฐ ๋์ ์คํ(Real-Time Interrupt-driven Concurrency)โ
- ๊ณต์ ๋ฆฌ์์ค ๊ด๋ฆฌ, ๋ฉ์์ง ์ ๋ฌ, ํ์คํฌ ์ค์ผ์ค๋ง, ํ์ด๋จธ ๋๊ธฐ์ด ์ง์
- Embassy
- ์ฐ์ ์์, ํ์ด๋จธ, ๋คํธ์ํน, USB๊ฐ ํฌํจ๋
async
์คํ์
- ์ฐ์ ์์, ํ์ด๋จธ, ๋คํธ์ํน, USB๊ฐ ํฌํจ๋
- TockOS
- ์ ์ ํ ์ค์ผ์ค๋ง ๋ฐ MMU๋ฅผ ์ง์ํ๋, ๋ณด์์ ์ค์ ์ ๋ ์ค์๊ฐ ์ด์์ฒด์
- Hubris
- Oxide Computer Company์์ ๋ง๋ ๋ง์ดํฌ๋ก์ปค๋ ๊ธฐ๋ฐ ์ค์๊ฐ ์ด์์ฒด์ ๋ก, ๋ฉ๋ชจ๋ฆฌ ๋ณดํธ, ๊ถํ์ด ์์ด ์ํ๋๋ ๋๋ผ์ด๋ฒ ๋ฑ์ ์ง์ํจ.
- FreeRTOS์ฉ ๋ฐ์ธ๋ฉ
std
๊ฐ ๊ตฌํ๋ ํ๋ซํผ๋ ์์ต๋๋ค(์: esp-idf).
- RTIC๋ ์ค์๊ฐ ์ด์์ฒด์ ๋ก ๋ณผ ์๋ ์๊ณ , ๋์์ฑ ์ง์์ ์ํ ํ๋ ์์ํฌ๋ก ๋ณผ ์๋ ์์ต๋๋ค.
- HAL์ ํฌํจํ์ง๋ ์์ต๋๋ค.
- ์ค์ผ์ค๋ง์ ์ปค๋์ด ์๋๋ผ Cortex-M NVIC(Nested Virtual Interrupt Controller)๋ก ๊ตฌํ์ด ๋ฉ๋๋ค.
- Cortex-M ์ ์ฉ์ ๋๋ค.
- Google์์๋ Titan ๋ณด์ ํค์ ์ฌ์ฉ๋๋ Haven ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์์ TockOS๋ฅผ ์ฌ์ฉํฉ๋๋ค.
- FreeRTOS๋ ๋๋ถ๋ถ C๋ก ์์ฑ๋์ง๋ง, ์ ํ๋ฆฌ์ผ์ด์ ์ Rust๋ก ์์ฑํ ์ ์๋๋ก ํด ์ฃผ๋ Rust ๋ฐ์ธ๋ฉ์ด ์ ๊ณต๋ฉ๋๋ค.
์ฐ์ต๋ฌธ์
We will read the direction from an I2C compass, and log the readings to a serial port.
After looking at the exercises, you can look at the solutions provided.
๋์นจ๋ฐ
We will read the direction from an I2C compass, and log the readings to a serial port. If you have time, try displaying it on the LEDs somehow too, or use the buttons somehow.
Hints:
- Check the documentation for the
lsm303agr
andmicrobit-v2
crates, as well as the micro:bit hardware. - The LSM303AGR Inertial Measurement Unit is connected to the internal I2C bus.
- TWI is another name for I2C, so the I2C master peripheral is called TWIM.
- The LSM303AGR driver needs something implementing the
embedded_hal::blocking::i2c::WriteRead
trait. Themicrobit::hal::Twim
struct implements this. - You have a
microbit::Board
struct with fields for the various pins and peripherals. - You can also look at the nRF52833 datasheet if you want, but it shouldnโt be necessary for this exercise.
Download the exercise template and look in the compass
directory for the following files.
src/main.rs
:
#![no_main] #![no_std] extern crate panic_halt as _; use core::fmt::Write; use cortex_m_rt::entry; use microbit::{hal::uarte::{Baudrate, Parity, Uarte}, Board}; #[entry] fn main() -> ! { let board = Board::take().unwrap(); // Configure serial port. let mut serial = Uarte::new( board.UARTE0, board.uart.into(), Parity::EXCLUDED, Baudrate::BAUD115200, ); // Set up the I2C controller and Inertial Measurement Unit. // TODO writeln!(serial, "Ready.").unwrap(); loop { // Read compass data and log it to the serial port. // TODO } }
Cargo.toml
(you shouldnโt need to change this):
[workspace]
[package]
name = "compass"
version = "0.1.0"
edition = "2021"
publish = false
[dependencies]
cortex-m-rt = "0.7.3"
embedded-hal = "0.2.6"
lsm303agr = "0.2.2"
microbit-v2 = "0.13.0"
panic-halt = "0.2.0"
Embed.toml
(you shouldnโt need to change this):
[default.general]
chip = "nrf52833_xxAA"
[debug.gdb]
enabled = true
[debug.reset]
halt_afterwards = true
.cargo/config.toml
(you shouldnโt need to change this):
[build]
target = "thumbv7em-none-eabihf" # Cortex-M4F
[target.'cfg(all(target_arch = "arm", target_os = "none"))']
rustflags = ["-C", "link-arg=-Tlink.x"]
See the serial output on Linux with:
picocom --baud 115200 --imap lfcrlf /dev/ttyACM0
Or on Mac OS something like (the device name may be slightly different):
picocom --baud 115200 --imap lfcrlf /dev/tty.usbmodem14502
Use Ctrl+A Ctrl+Q to quit picocom.
์ ํ๋ฆฌ์ผ์ด์ ํ๋ก์ธ์
์ง๊ธ๊น์ง Arm Cortex-M ์๋ฆฌ์ฆ์ ๊ฐ์ ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์ ๊ดํด ์์๋ดค์ต๋๋ค. ์ด์ ์ ํ๋ฆฌ์ผ์ด์ ํ๋ก์ธ์์ธ Cortex-A๋ฅผ ์ํ ์ฝ๋๋ฅผ ์์ฑํด ๋ณด๊ฒ ์ต๋๋ค. ํธ์์ QEMU์ aarch64 โvirtโ ๋ณด๋๋ฅผ ์ฌ์ฉํฉ๋๋ค.
- ์ผ๋ฐ์ ์ผ๋ก ๋ง์ดํฌ๋ก์ปจํธ๋กค๋ฌ์๋ MMU ๋๋ ๋ค์ค ๋ ๋ฒจ ๊ถํ(Arm CPU์์๋ ์ต์ ์ ๋ ๋ฒจ(exception level), x86์์๋ ๋ง(ring))์ด ์์ต๋๋ค. ์ ํ๋ฆฌ์ผ์ด์ ํ๋ก์ธ์๋ ์ด๋ค์ ๊ฐ์ง๊ณ ์์ต๋๋ค.
- QEMU๋ ์ํคํ ์ฒ๋ณ๋ก ๋ค์ํ ๋จธ์ ๋๋ ๋ณด๋ ๋ชจ๋ธ์ ์๋ฎฌ๋ ์ด์ ํ ์ ์์ต๋๋ค. โvirtโ ๋ณด๋๋ ํน์ ์ค์ ํ๋์จ์ด๋ฅผ ์๋ฎฌ๋ ์ด์ ํ์ง ์์ผ๋ฉฐ, ๊ฐ์ ๋จธ์ ์ฉ์ผ๋ก๋ง ์ค๊ณ๋์์ต๋๋ค.
Rust ์ํ ์ค๋น
Before we can start running Rust code, we need to do some initialisation.
.section .init.entry, "ax"
.global entry
entry:
/*
* Load and apply the memory management configuration, ready to enable MMU and
* caches.
*/
adrp x30, idmap
msr ttbr0_el1, x30
mov_i x30, .Lmairval
msr mair_el1, x30
mov_i x30, .Ltcrval
/* Copy the supported PA range into TCR_EL1.IPS. */
mrs x29, id_aa64mmfr0_el1
bfi x30, x29, #32, #4
msr tcr_el1, x30
mov_i x30, .Lsctlrval
/*
* Ensure everything before this point has completed, then invalidate any
* potentially stale local TLB entries before they start being used.
*/
isb
tlbi vmalle1
ic iallu
dsb nsh
isb
/*
* Configure sctlr_el1 to enable MMU and cache and don't proceed until this
* has completed.
*/
msr sctlr_el1, x30
isb
/* Disable trapping floating point access in EL1. */
mrs x30, cpacr_el1
orr x30, x30, #(0x3 << 20)
msr cpacr_el1, x30
isb
/* Zero out the bss section. */
adr_l x29, bss_begin
adr_l x30, bss_end
0: cmp x29, x30
b.hs 1f
stp xzr, xzr, [x29], #16
b 0b
1: /* Prepare the stack. */
adr_l x30, boot_stack_end
mov sp, x30
/* Set up exception vector. */
adr x30, vector_table_el1
msr vbar_el1, x30
/* Call into Rust code. */
bl main
/* Loop forever waiting for interrupts. */
2: wfi
b 2b
- This is the same as it would be for C: initialising the processor state, zeroing the BSS, and setting up the stack pointer.
- The BSS (block starting symbol, for historical reasons) is the part of the object file which containing statically allocated variables which are initialised to zero. They are omitted from the image, to avoid wasting space on zeroes. The compiler assumes that the loader will take care of zeroing them.
- The BSS may already be zeroed, depending on how memory is initialised and the image is loaded, but we zero it to be sure.
- We need to enable the MMU and cache before reading or writing any memory. If we donโt:
- Unaligned accesses will fault. We build the Rust code for the
aarch64-unknown-none
target which sets+strict-align
to prevent the compiler generating unaligned accesses, so it should be fine in this case, but this is not necessarily the case in general. - If it were running in a VM, this can lead to cache coherency issues. The problem is that the VM is accessing memory directly with the cache disabled, while the host has cacheable aliases to the same memory. Even if the host doesnโt explicitly access the memory, speculative accesses can lead to cache fills, and then changes from one or the other will get lost when the cache is cleaned or the VM enables the cache. (Cache is keyed by physical address, not VA or IPA.)
- Unaligned accesses will fault. We build the Rust code for the
- For simplicity, we just use a hardcoded pagetable (see
idmap.S
) which identity maps the first 1 GiB of address space for devices, the next 1 GiB for DRAM, and another 1 GiB higher up for more devices. This matches the memory layout that QEMU uses. - We also set up the exception vector (
vbar_el1
), which weโll see more about later. - All examples this afternoon assume we will be running at exception level 1 (EL1). If you need to run at a different exception level youโll need to modify
entry.S
accordingly.
์ธ๋ผ์ธ ์ด์ ๋ธ๋ฆฌ
๊ฐ๋ Rust ์ฝ๋๋ก๋ ๊ตฌํ์ด ๋ถ๊ฐ๋ฅํ ์์ ๋ค์ด ์์ผ๋ฉฐ, ์ด ๊ฒฝ์ฐ ์ด์ ๋ธ๋ฆฌ๋ฅผ ์ฌ์ฉํด์ผ ํฉ๋๋ค. ์๋ฅผ ๋ค์ด ํ์จ์ด๋ฅผ ํฅํด์ ์์คํ ์ ์์ ๋๋ผ๊ณ HVC๋ฅผ ํธ์ถํ๋ ๋ฐฉ๋ฒ์ ๋ค์๊ณผ ๊ฐ์ต๋๋ค.
#![no_main] #![no_std] use core::arch::asm; use core::panic::PanicInfo; mod exceptions; const PSCI_SYSTEM_OFF: u32 = 0x84000008; #[no_mangle] extern "C" fn main(_x0: u64, _x1: u64, _x2: u64, _x3: u64) { // Safe because this only uses the declared registers and doesn't do // anything with memory. unsafe { asm!("hvc #0", inout("w0") PSCI_SYSTEM_OFF => _, inout("w1") 0 => _, inout("w2") 0 => _, inout("w3") 0 => _, inout("w4") 0 => _, inout("w5") 0 => _, inout("w6") 0 => _, inout("w7") 0 => _, options(nomem, nostack) ); } loop {} }
์ค์ ๋ก ์ด๋ฅผ ์คํํ๋ ค๋ฉด ์ด๋ฌํ ๋ชจ๋ ํจ์๋ฅผ ์ํ ๋ํผ๊ฐ ํฌํจ๋ smccc
ํฌ๋ ์ดํธ๋ฅผ ์ฌ์ฉํ์ธ์.
- PSCI (Power State Coordination Interface)๋ ์์คํ ๋ฐ CPU ์ ์ ์ํ๋ฅผ ๊ด๋ฆฌํ๋ Arm์ ํ์ค ์ธํฐํ์ด์ค์ ๋๋ค. ์ด ์ธํฐํ์ด์ค๋ EL3 ํ์จ์ด์ ํ์ดํผ๋ฐ์ด์ ์ ์ํด ๊ตฌํ๋ฉ๋๋ค.
0 => _
๋ฌธ๋ฒ์ ์ธ๋ผ์ธ ์ด์ ๋ธ๋ฆฌ ์ฝ๋๋ฅผ ์คํํ๊ธฐ ์ ์ ๋ ์ง์คํฐ๋ฅผ 0์ผ๋ก ์ด๊ธฐํํ๊ณ ๊ทธ ํ์๋ ๊ทธ ๋ ์ง์คํฐ์ ๊ฐ์ ๋ฌด์ํจ์ ์๋ฏธํฉ๋๋ค. ํธ์ถ ์ ๋ ์ง์คํฐ์ ๊ฐ์ด ๋ฎ์ด ์จ์ง ์ ์์ผ๋ฏ๋กin
๋์inout
์ ์ฌ์ฉํด์ผ ํฉ๋๋ค.- ์ด
main
ํจ์๋#[no_mangle]
๋ฐextern "C"
์ฌ์ผ ํฉ๋๋ค. ์๋ํ๋ฉด ์ด ํจ์๋ Rust ์ฝ๋๊ฐ ์๋, ์ด์ ๋ธ๋ฌ๋ก ์์ฑ๋entry.S
์์ ํธ์ถ๋๊ธฐ ๋๋ฌธ์ ๋๋ค. _x0
โ_x3
๋x0
์์x3
๋ ์ง์คํฐ๋ค์ ๊ฐ์ ๋๋ค. ์ด ๋ ์ง์คํฐ๋ค์ ์ผ๋ฐ์ ์ผ๋ก ๋ถํธ๋ก๋์์ ๋๋ฐ์ด์ค ํธ๋ฆฌ์ ๋ํ ํฌ์ธํฐ ๋ฑ์ ์ ๋ฌํ ๋ ์ฌ์ฉ๋ฉ๋๋ค. ํ์ค aarch64 ํธ์ถ ๊ท์ฝ(extern "C"
์์ ์ฌ์ฉํ๋๋ก ์ง์ )์ ๋ฐ๋ผ ๋ ์ง์คํฐx0
์์x7
์ด ํจ์์ ์ ๋ฌ๋ ์ฒ์ 8๊ฐ ์ธ์์ ์ฌ์ฉ๋๋ฏ๋กentry.S
๋ ์ด๋ฌํ ๋ ์ง์คํฐ๋ฅผ ๋ณ๊ฒฝํ์ง ์๋์ง ํ์ธํ๋ ๊ฒ ์ธ์๋ ํน๋ณํ ํ ์์ ์ด ์์ต๋๋ค.src/bare-metal/aps/examples
์์make qemu_psci
๋ฅผ ์ฌ์ฉํ์ฌ QEMU์์ ์์๋ฅผ ์คํํฉ๋๋ค.
MMIO๋ฅผ ์ํ ํ๋ฐ์ฑ(volatile) ๋ฉ๋ชจ๋ฆฌ ์ก์ธ์ค
pointer::read_volatile
๋ฐpointer::write_volatile
์ ์ฌ์ฉํ์ธ์.- ์ฐธ์กฐ๋ฅผ ์ ์งํ์ง ๋ง์ธ์.
addr_of!
๋ฅผ ์ฌ์ฉํ๋ฉด ์์ ์ฉ๋์ ์ฐธ์กฐ๋ฅผ ๋ง๋ค์ง ์๊ณ ๋ ๊ตฌ์กฐ์ฒด ํ๋๋ฅผ ๊ฐ์ ธ์ฌ ์ ์์ต๋๋ค.
- ํ๋ฐ์ฑ(volatile) ์ก์ธ์ค: ์ฝ๊ธฐ ๋๋ ์ฐ๊ธฐ ์์
์ด ๋ถ์ ํจ๊ณผ(side effect)๋ฅผ ๋๋ฐํ ์ ์๊ธฐ ๋๋ฌธ์ ์ปดํ์ผ๋ฌ๋ ํ๋์จ์ด๊ฐ ์์๋ก ์ด๋ฅผ ์ฝ๊ธฐ ์ฐ๊ธฐ ์์
์ ์์๋ฅผ ๋ฐ๊พธ๊ฑฐ๋, ์ค๋ณตํด์ ์ํํ๊ฑฐ๋ ๋๋ ์ ๊ฑฐํ์ง ๋ชปํ๊ฒ ํฉ๋๋ค.
- ์ผ๋ฐ์ ์ผ๋ก ์ฐ๊ณ ๋ ํ ์ฝ์ผ๋ฉด(์: ๋ณ๊ฒฝ ๊ฐ๋ฅํ ์ฐธ์กฐ๋ฅผ ํตํด) ์ปดํ์ผ๋ฌ๋ ์ฝ์ ๊ฐ์ด ๋ฐฉ๊ธ ์ด ๊ฐ๊ณผ ๋์ผํ๋ค๊ณ ๊ฐ์ ํ๊ณ ์ค์ ๋ก ๋ฉ๋ชจ๋ฆฌ๋ฅผ ์ฝ์ง ์์ ์ ์์ต๋๋ค.
- ํ๋์จ์ด์ ๋ํ ํ๋ฐ์ฑ ์ก์ธ์ค๋ฅผ ์ํ ์ผ๋ถ ๊ธฐ์กด ํฌ๋ ์ดํธ๋ ์ฐธ์กฐ๋ฅผ ์ ์งํ์ง๋ง ์ด๋ ์ฌ๋ฐ๋ฅธ ๊ฒ์ด ์๋๋๋ค. ์ฐธ์กฐ๊ฐ ์์ ๋๋ง๋ค ์ปดํ์ผ๋ฌ๋ ์ด๋ฅผ ์ญ์ฐธ์กฐํ๋๋ก ์ ํํ ์ ์์ต๋๋ค.
addr_of!
๋งคํฌ๋ก๋ฅผ ์ฌ์ฉํ์ฌ ๊ตฌ์กฐ์ฒด ํฌ์ธํฐ์์ ๊ตฌ์กฐ์ฒด ํ๋ ํฌ์ธํฐ๋ฅผ ๊ฐ์ ธ์ต๋๋ค.
UART ๋๋ผ์ด๋ฒ ์์ฑ
QEMU์ โvirtโ ๋ณด๋์๋ PL011 UART๊ฐ ์์ผ๋ฏ๋ก ์ด๋ฅผ ์ํ ๋๋ผ์ด๋ฒ๋ฅผ ์์ฑํด ๋ณด๊ฒ ์ต๋๋ค.
const FLAG_REGISTER_OFFSET: usize = 0x18; const FR_BUSY: u8 = 1 << 3; const FR_TXFF: u8 = 1 << 5; /// Minimal driver for a PL011 UART. #[derive(Debug)] pub struct Uart { base_address: *mut u8, } impl Uart { /// Constructs a new instance of the UART driver for a PL011 device at the /// given base address. /// /// # Safety /// /// The given base address must point to the 8 MMIO control registers of a /// PL011 device, which must be mapped into the address space of the process /// as device memory and not have any other aliases. pub unsafe fn new(base_address: *mut u8) -> Self { Self { base_address } } /// Writes a single byte to the UART. pub fn write_byte(&self, byte: u8) { // Wait until there is room in the TX buffer. while self.read_flag_register() & FR_TXFF != 0 {} // Safe because we know that the base address points to the control // registers of a PL011 device which is appropriately mapped. unsafe { // Write to the TX buffer. self.base_address.write_volatile(byte); } // Wait until the UART is no longer busy. while self.read_flag_register() & FR_BUSY != 0 {} } fn read_flag_register(&self) -> u8 { // Safe because we know that the base address points to the control // registers of a PL011 device which is appropriately mapped. unsafe { self.base_address.add(FLAG_REGISTER_OFFSET).read_volatile() } } }
Uart::new
๋ ์์ ํ์ง ์์ง๋ง(usafe), ๊ทธ ์ธ ๋ค๋ฅธ ๋ฉ์๋๋ค์ ์์ ํ(safe) ์ ์ ์ฃผ๋ชฉํ์ธ์.๋ค๋ฅธ ๋ฉ์๋๋ค์ด ์์ ํ ์ ์๋ ์ด์ ๋,Uart::new
์ ์์ ์๊ตฌ์ฌํญ(์ฆ, ์ง์ ๋ UART์ ๋๋ผ์ด๋ฒ ์ธ์คํด์ค๊ฐ ํ๋๋ง ์์ผ๋ฉฐ ์ฃผ์ ๊ณต๊ฐ์ ๋ณ์นญ์ ์ง์ ํ๋ ๋ค๋ฅธ ํญ๋ชฉ์ด ์์) ์ด ๋ง์กฑ๋๊ธฐ๋ง ํ๋ฉดwrite_byte
์ ๊ฐ์ ํจ์๋ฅผ ์์ ํ๊ฒ ํธ์ถํ๋๋ฐ ์์ด์ ํ์ํ ๋ชจ๋ ์ ์ ์กฐ๊ฑด์ด ๋ง์กฑ๋๊ธฐ ๋๋ฌธ์ ๋๋ค.- ๋ฐ๋ ๋ฐฉ๋ฒ์ผ๋ก๋ ์คํํ ์ ์์ง๋ง(
new
๋ฅผ ์์ ํ๊ฒ ๋ง๋ค๊ณwrite_byte
๋ฅผ ์์ ํ์ง ์๊ฒ ๋ง๋ฆ) ์ด๋write_byte
๋ฅผ ํธ์ถํ๋ ๋ชจ๋ ์์น์์ ์์ ์ฑ์ ๊ดํด ์ถ๋ก ํด์ผ ํ๋ฏ๋ก ์ฌ์ฉ ํธ์์ฑ์ด ํจ์ฌ ๋จ์ด์ง๋๋ค. - ์ด๋ ์์ ํ์ง ์์ ์ฝ๋์ ์์ ํ ๋ํผ๋ฅผ ์์ฑํ๋ ์ผ๋ฐ์ ์ธ ํจํด์ ๋๋ค. ์์ ์ ๊ดํ ์ฆ๋ช ๋ถ๋ด์ ์ฌ๋ฌ ๋ง์ ์์น์์ ์์์ ์์น๋ก ์ฎ๊ธฐ๋ ๊ฒ์ ๋๋ค.
More traits
We derived the Debug
trait. It would be useful to implement a few more traits too.
use core::fmt::{self, Write}; impl Write for Uart { fn write_str(&mut self, s: &str) -> fmt::Result { for c in s.as_bytes() { self.write_byte(*c); } Ok(()) } } // Safe because it just contains a pointer to device memory, which can be // accessed from any context. unsafe impl Send for Uart {}
- Implementing
Write
lets us use thewrite!
andwriteln!
macros with ourUart
type. - Run the example in QEMU with
make qemu_minimal
undersrc/bare-metal/aps/examples
.
๋ ๋์ UART ๋๋ผ์ด๋ฒ
PL011์๋ ์ค์ ๋ก ํจ์ฌ ๋ ๋ง์ ๋ ์ง์คํฐ๊ฐ ์์ผ๋ฉฐ, ์ด์ ์ก์ธ์คํ ํฌ์ธํฐ๋ฅผ ๊ตฌ์ฑํ๊ธฐ ์ํด ์คํ์ ์ ์ถ๊ฐํ๋ฉด ์ค๋ฅ๊ฐ ๋ฐ์ํ๊ธฐ ์ฝ๊ณ ์ฝ๊ธฐ ์ด๋ ต์ต๋๋ค. ๋ํ ๊ทธ์ค ์ผ๋ถ๋ ๊ตฌ์กฐํ๋ ๋ฐฉ์์ผ๋ก ์ก์ธ์คํ ์ ์๋ ๋นํธ ํ๋์ ๋๋ค.
์คํ์ | ๋ ์ง์คํฐ ์ด๋ฆ | ๋๋น |
---|---|---|
0x00 | DR | 12 |
0x04 | RSR | 4 |
0x18 | FR | 9 |
0x20 | ILPR | 8 |
0x24 | IBRD | 16 |
0x28 | FBRD | 6 |
0x2c | LCR_H | 8 |
0x30 | CR | 16 |
0x34 | IFLS | 6 |
0x38 | IMSC | 11 |
0x3c | RIS | 11 |
0x40 | MIS | 11 |
0x44 | ICR | 11 |
0x48 | DMACR | 3 |
- ๊ฐ๊ฒฐ์ฑ์ ์ํด ์ผ๋ถ ID ๋ ์ง์คํฐ๋ ์๋ต๋์์ต๋๋ค.
๋นํธํ๋๊ทธ
The bitflags
crate is useful for working with bitflags.
use bitflags::bitflags; bitflags! { /// Flags from the UART flag register. #[repr(transparent)] #[derive(Copy, Clone, Debug, Eq, PartialEq)] struct Flags: u16 { /// Clear to send. const CTS = 1 << 0; /// Data set ready. const DSR = 1 << 1; /// Data carrier detect. const DCD = 1 << 2; /// UART busy transmitting data. const BUSY = 1 << 3; /// Receive FIFO is empty. const RXFE = 1 << 4; /// Transmit FIFO is full. const TXFF = 1 << 5; /// Receive FIFO is full. const RXFF = 1 << 6; /// Transmit FIFO is empty. const TXFE = 1 << 7; /// Ring indicator. const RI = 1 << 8; } }
- The
bitflags!
macro creates a newtype something likeFlags(u16)
, along with a bunch of method implementations to get and set flags.
Multiple registers
We can use a struct to represent the memory layout of the UARTโs registers.
#[repr(C, align(4))] struct Registers { dr: u16, _reserved0: [u8; 2], rsr: ReceiveStatus, _reserved1: [u8; 19], fr: Flags, _reserved2: [u8; 6], ilpr: u8, _reserved3: [u8; 3], ibrd: u16, _reserved4: [u8; 2], fbrd: u8, _reserved5: [u8; 3], lcr_h: u8, _reserved6: [u8; 3], cr: u16, _reserved7: [u8; 3], ifls: u8, _reserved8: [u8; 3], imsc: u16, _reserved9: [u8; 2], ris: u16, _reserved10: [u8; 2], mis: u16, _reserved11: [u8; 2], icr: u16, _reserved12: [u8; 2], dmacr: u8, _reserved13: [u8; 3], }
#[repr(C)]
tells the compiler to lay the struct fields out in order, following the same rules as C. This is necessary for our struct to have a predictable layout, as default Rust representation allows the compiler to (among other things) reorder fields however it sees fit.
๋๋ผ์ด๋ฒ
Now letโs use the new Registers
struct in our driver.
/// Driver for a PL011 UART. #[derive(Debug)] pub struct Uart { registers: *mut Registers, } impl Uart { /// Constructs a new instance of the UART driver for a PL011 device at the /// given base address. /// /// # Safety /// /// The given base address must point to the 8 MMIO control registers of a /// PL011 device, which must be mapped into the address space of the process /// as device memory and not have any other aliases. pub unsafe fn new(base_address: *mut u32) -> Self { Self { registers: base_address as *mut Registers, } } /// Writes a single byte to the UART. pub fn write_byte(&self, byte: u8) { // Wait until there is room in the TX buffer. while self.read_flag_register().contains(Flags::TXFF) {} // Safe because we know that self.registers points to the control // registers of a PL011 device which is appropriately mapped. unsafe { // Write to the TX buffer. addr_of_mut!((*self.registers).dr).write_volatile(byte.into()); } // Wait until the UART is no longer busy. while self.read_flag_register().contains(Flags::BUSY) {} } /// Reads and returns a pending byte, or `None` if nothing has been received. pub fn read_byte(&self) -> Option<u8> { if self.read_flag_register().contains(Flags::RXFE) { None } else { let data = unsafe { addr_of!((*self.registers).dr).read_volatile() }; // TODO: Check for error conditions in bits 8-11. Some(data as u8) } } fn read_flag_register(&self) -> Flags { // Safe because we know that self.registers points to the control // registers of a PL011 device which is appropriately mapped. unsafe { addr_of!((*self.registers).fr).read_volatile() } } }
- Note the use of
addr_of!
/addr_of_mut!
to get pointers to individual fields without creating an intermediate reference, which would be unsound.
Using it
Letโs write a small program using our driver to write to the serial console, and echo incoming bytes.
#![no_main] #![no_std] mod exceptions; mod pl011; use crate::pl011::Uart; use core::fmt::Write; use core::panic::PanicInfo; use log::error; use smccc::psci::system_off; use smccc::Hvc; /// Base address of the primary PL011 UART. const PL011_BASE_ADDRESS: *mut u32 = 0x900_0000 as _; #[no_mangle] extern "C" fn main(x0: u64, x1: u64, x2: u64, x3: u64) { // Safe because `PL011_BASE_ADDRESS` is the base address of a PL011 device, // and nothing else accesses that address range. let mut uart = unsafe { Uart::new(PL011_BASE_ADDRESS) }; writeln!(uart, "main({x0:#x}, {x1:#x}, {x2:#x}, {x3:#x})").unwrap(); loop { if let Some(byte) = uart.read_byte() { uart.write_byte(byte); match byte { b'\r' => { uart.write_byte(b'\n'); } b'q' => break, _ => {} } } } writeln!(uart, "Bye!").unwrap(); system_off::<Hvc>().unwrap(); }
- As in the inline assembly example, this
main
function is called from our entry point code inentry.S
. See the speaker notes there for details. - Run the example in QEMU with
make qemu
undersrc/bare-metal/aps/examples
.
๋ก๊น
log
ํฌ๋ ์ดํธ์ ๋ก๊น
๋งคํฌ๋ก๋ฅผ ์ฌ์ฉํ ์ ์์ผ๋ฉด ์ข์ต๋๋ค. ์ด๋ Log
ํธ๋ ์์ ๊ตฌํํ๋ฉด ๋ฉ๋๋ค.
use crate::pl011::Uart; use core::fmt::Write; use log::{LevelFilter, Log, Metadata, Record, SetLoggerError}; use spin::mutex::SpinMutex; static LOGGER: Logger = Logger { uart: SpinMutex::new(None), }; struct Logger { uart: SpinMutex<Option<Uart>>, } impl Log for Logger { fn enabled(&self, _metadata: &Metadata) -> bool { true } fn log(&self, record: &Record) { writeln!( self.uart.lock().as_mut().unwrap(), "[{}] {}", record.level(), record.args() ) .unwrap(); } fn flush(&self) {} } /// Initialises UART logger. pub fn init(uart: Uart, max_level: LevelFilter) -> Result<(), SetLoggerError> { LOGGER.uart.lock().replace(uart); log::set_logger(&LOGGER)?; log::set_max_level(max_level); Ok(()) }
log
ํจ์ ์์์unwrap
ํ๋ ๊ฒ์ ๊ด์ฐฎ์ต๋๋ค.. ์๋ํ๋ฉดset_logger
๋ฅผ ํธ์ถํ๊ธฐ ์ ์LOGGER
๋ฅผ ์ด๊ธฐํํ๊ธฐ ๋๋ฌธ์ ๋๋ค.
Using it
We need to initialise the logger before we use it.
#![no_main] #![no_std] mod exceptions; mod logger; mod pl011; use crate::pl011::Uart; use core::panic::PanicInfo; use log::{error, info, LevelFilter}; use smccc::psci::system_off; use smccc::Hvc; /// Base address of the primary PL011 UART. const PL011_BASE_ADDRESS: *mut u32 = 0x900_0000 as _; #[no_mangle] extern "C" fn main(x0: u64, x1: u64, x2: u64, x3: u64) { // Safe because `PL011_BASE_ADDRESS` is the base address of a PL011 device, // and nothing else accesses that address range. let uart = unsafe { Uart::new(PL011_BASE_ADDRESS) }; logger::init(uart, LevelFilter::Trace).unwrap(); info!("main({x0:#x}, {x1:#x}, {x2:#x}, {x3:#x})"); assert_eq!(x1, 42); system_off::<Hvc>().unwrap(); } #[panic_handler] fn panic(info: &PanicInfo) -> ! { error!("{info}"); system_off::<Hvc>().unwrap(); loop {} }
- Note that our panic handler can now log details of panics.
- Run the example in QEMU with
make qemu_logger
undersrc/bare-metal/aps/examples
.
์์ธ
AArch64 defines an exception vector table with 16 entries, for 4 types of exceptions (synchronous, IRQ, FIQ, SError) from 4 states (current EL with SP0, current EL with SPx, lower EL using AArch64, lower EL using AArch32). We implement this in assembly to save volatile registers to the stack before calling into Rust code:
use log::error; use smccc::psci::system_off; use smccc::Hvc; #[no_mangle] extern "C" fn sync_exception_current(_elr: u64, _spsr: u64) { error!("sync_exception_current"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn irq_current(_elr: u64, _spsr: u64) { error!("irq_current"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn fiq_current(_elr: u64, _spsr: u64) { error!("fiq_current"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn serr_current(_elr: u64, _spsr: u64) { error!("serr_current"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn sync_lower(_elr: u64, _spsr: u64) { error!("sync_lower"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn irq_lower(_elr: u64, _spsr: u64) { error!("irq_lower"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn fiq_lower(_elr: u64, _spsr: u64) { error!("fiq_lower"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn serr_lower(_elr: u64, _spsr: u64) { error!("serr_lower"); system_off::<Hvc>().unwrap(); }
- EL is exception level; all our examples this afternoon run in EL1.
- For simplicity we arenโt distinguishing between SP0 and SPx for the current EL exceptions, or between AArch32 and AArch64 for the lower EL exceptions.
- For this example we just log the exception and power down, as we donโt expect any of them to actually happen.
- We can think of exception handlers and our main execution context more or less like different threads.
Send
andSync
will control what we can share between them, just like with threads. For example, if we want to share some value between exception handlers and the rest of the program, and itโsSend
but notSync
, then weโll need to wrap it in something like aMutex
and put it in a static.
๋ค๋ฅธ ํ๋ก์ ํธ
- oreboot
- โcoreboot without the Cโ
- Supports x86, aarch64 and RISC-V.
- Relies on LinuxBoot rather than having many drivers itself.
- Rust RaspberryPi OS tutorial
- Initialisation, UART driver, simple bootloader, JTAG, exception levels, exception handling, page tables
- Some dodginess around cache maintenance and initialisation in Rust, not necessarily a good example to copy for production code.
cargo-call-stack
- Static analysis to determine maximum stack usage.
- The RaspberryPi OS tutorial runs Rust code before the MMU and caches are enabled. This will read and write memory (e.g. the stack). However:
- Without the MMU and cache, unaligned accesses will fault. It builds with
aarch64-unknown-none
which sets+strict-align
to prevent the compiler generating unaligned accesses so it should be alright, but this is not necessarily the case in general. - If it were running in a VM, this can lead to cache coherency issues. The problem is that the VM is accessing memory directly with the cache disabled, while the host has cacheable aliases to the same memory. Even if the host doesnโt explicitly access the memory, speculative accesses can lead to cache fills, and then changes from one or the other will get lost. Again this is alright in this particular case (running directly on the hardware with no hypervisor), but isnโt a good pattern in general.
- Without the MMU and cache, unaligned accesses will fault. It builds with
์ ์ฉํ ํฌ๋ ์ดํธ
bare-metal ํ๋ก๊ทธ๋๋ฐ์ ๋ช ๊ฐ์ง ์ผ๋ฐ์ ์ธ ๋ฌธ์ ๋ฅผ ํด๊ฒฐํ๋ ํฌ๋ ์ดํธ๋ฅผ ์ดํด๋ด ๋๋ค.
zerocopy
Fuchsiaํ์ด ๋ง๋ zerocopy
ํฌ๋ ์ดํธ๋ ๋ฐ์ดํธ ์ํ์ค๋ฅผ ๋ค๋ฅธ ํ์
์ผ๋ก ์์ ํ๊ฒ ๋ณํํ๊ธฐ ์ํ ํธ๋ ์ ๋ฐ ๋งคํฌ๋ก๋ฅผ ์ ๊ณตํฉ๋๋ค.
use zerocopy::AsBytes; #[repr(u32)] #[derive(AsBytes, Debug, Default)] enum RequestType { #[default] In = 0, Out = 1, Flush = 4, } #[repr(C)] #[derive(AsBytes, Debug, Default)] struct VirtioBlockRequest { request_type: RequestType, reserved: u32, sector: u64, } fn main() { let request = VirtioBlockRequest { request_type: RequestType::Flush, sector: 42, ..Default::default() }; assert_eq!( request.as_bytes(), &[4, 0, 0, 0, 0, 0, 0, 0, 42, 0, 0, 0, 0, 0, 0, 0] ); }
์ด ํฌ๋ ์ดํธ๋ ํ๋ฐ์ฑ(volatile) ์ฝ๊ธฐ ๋ฐ ์ฐ๊ธฐ๋ฅผ ์ฌ์ฉํ์ง ์์ผ๋ฏ๋ก MMIO์ ์ ํฉํ์ง ์์ง๋ง, ํ๋์จ์ด์ ๊ณต์ ๋๊ฑฐ๋(์: DMA์์) ์ธ์ฅ ์ธํฐํ์ด์ค๋ฅผ ํตํด ์ ์ก๋๋ ๊ตฌ์กฐ์ฒด๋ฅผ ๋ค๋ฃจ๋ ๋ฐ์๋ ์ ์ฉํ ์ ์์ต๋๋ค.
- ์ด๋ค ํ์
์ด ๊ฐ๋ฅํ ๋ชจ๋ ๋ฐ์ดํธ ํจํด๋ค์ ๋ํด ์ฌ๋ฐ๋ฅธ ๊ฐ์ ๊ฐ์ง ๋์๋ง , ๊ทธ ํ์
์ด
FromBytes
๋ฅผ ๊ตฌํํ ์ ์์ต๋๋ค. ๊ทธ๋ ๊ฒ ํด์ ์ ๋ขฐํ ์ ์๋ ๋ฐ์ดํธ ์ํ์ค๋ฅผ ์์ ํ๊ฒ ํด๋น ํ์ ์ผ๋ก ๋ณํํ ์ ์์ต๋๋ค. - ์ ์ฝ๋์์ ์ ์ํ ํ์
์ ๋ํด
FromBytes
๋ฅผ ๊ตฌํํ๋ ค๊ณ ํ๋ฉด ์๋ฌ๊ฐ ๋ฐ์ํฉ๋๋ค.RequestType
์ ๊ฐ๋ฅํ ๋ชจ๋ u32 ๊ฐ์ ์๋ณ์๋ก ๋ฐ์๋ค์ด์ง ์๊ธฐ ๋๋ฌธ์ ๋๋ค. ์ฆ ๋ชจ๋ ๋ฐ์ดํธ ํจํด์ด ์ ํจํRequestType
๊ฐ์ ์๋๋๋ค. zerocopy::byteorder
์๋ ๋ฐ์ดํธ ์ค๋์ ๋ฐ๋ฅธ ์๋ก ๋ค๋ฅธ ํํ ๋ฐฉ์์ ์ง์ํ๋ ์ซ์ ํ์ ์ ์ ๊ณตํฉ๋๋ค.src/bare-metal/useful-crates/zerocopy-example/
์์cargo run
์ ์ฌ์ฉํ์ฌ ์์๋ฅผ ์คํํฉ๋๋ค(์ข ์์ฑ ๋ฌธ์ ๋ก ์ธํด ํ๋ ์ด๊ทธ๋ผ์ด๋์์๋ ์คํ๋์ง ์์ต๋๋ค).
aarch64-paging
aarch64-paging
ํฌ๋ ์ดํธ๋ฅผ ์ฌ์ฉํ๋ฉด AArch64 ๊ฐ์ ๋ฉ๋ชจ๋ฆฌ ์์คํ
์ํคํ
์ฒ์ ๋ฐ๋ผ ํ์ด์ง ํ
์ด๋ธ์ ๋ง๋ค ์ ์์ต๋๋ค.
use aarch64_paging::{ idmap::IdMap, paging::{Attributes, MemoryRegion}, }; const ASID: usize = 1; const ROOT_LEVEL: usize = 1; // Create a new page table with identity mapping. let mut idmap = IdMap::new(ASID, ROOT_LEVEL); // Map a 2 MiB region of memory as read-only. idmap.map_range( &MemoryRegion::new(0x80200000, 0x80400000), Attributes::NORMAL | Attributes::NON_GLOBAL | Attributes::READ_ONLY, ).unwrap(); // Set `TTBR0_EL1` to activate the page table. idmap.activate();
- ํ์ฌ๋ EL1๋ง ์ง์ํ์ง๋ง ๋ค๋ฅธ ์ต์ ์ ๋ ๋ฒจ(Exception Level: EL)๋ ์ด๋ ต์ง ์๊ฒ ์ถ๊ฐํ ์ ์์ต๋๋ค.
- Android์์ ๋ณดํธ๋ VM ํ์จ์ด์ ์ฌ์ฉ๋ฉ๋๋ค.
- ์ด ์์๋ฅผ ๊ฐ๋จํ๊ฒ ์คํํ๋ ๋ฐฉ๋ฒ์ ์์ต๋๋ค. ์ค์ ํ๋์จ์ด ๋๋ QEMU์์ ์คํํด์ผ ํ๊ธฐ ๋๋ฌธ์ ๋๋ค.
buddy_system_allocator
buddy_system_allocator
๋ ๋ฒ๋ ์์คํ
ํ ๋น์๋ฅผ ๊ตฌํํ๋ ์๋ ํํฐ ํฌ๋ ์ดํธ์
๋๋ค. ์ด ํฌ๋ ์ดํธ์ LockedHeap
์ GlobalAlloc
๋ฅผ ๊ตฌํํฉ๋๋ค. ๋ฐ๋ผ์ ์ฌ๋ฌ๋ถ์ ๋ฒ๋ ์์คํ
ํ ๋น์๋ฅผ โallocโ ํฌ๋ ์ดํธ๋ฅผ ํตํด์ ์ฌ์ฉํ ์ ์์ต๋๋ค(์ด์ ์ ํ์ธํจ). ๋๋ ๋ค๋ฅธ ์ฃผ์ ๊ณต๊ฐ์ ํ ๋นํ๋ ๋ฐ ์ฌ์ฉํ ์ ์์ต๋๋ค. ์๋ฅผ ๋ค์ด PCI BAR์ MMIO ๊ณต๊ฐ์ ํ ๋นํ ์ ์์ต๋๋ค.
use buddy_system_allocator::FrameAllocator; use core::alloc::Layout; fn main() { let mut allocator = FrameAllocator::<32>::new(); allocator.add_frame(0x200_0000, 0x400_0000); let layout = Layout::from_size_align(0x100, 0x100).unwrap(); let bar = allocator .alloc_aligned(layout) .expect("Failed to allocate 0x100 byte MMIO region"); println!("Allocated 0x100 byte MMIO region at {:#x}", bar); }
- PCI BAR๋ BAR์์ญ์ ํฌ๊ธฐ์ ๋ง์ถ์ด ์ ๋ ฌ๋ฉ๋๋ค.
src/bare-metal/useful-crates/allocator-example/
์์cargo run
์ ์ฌ์ฉํ์ฌ ์์๋ฅผ ์คํํฉ๋๋ค(์ข ์์ฑ ๋ฌธ์ ๋ก ์ธํด ํ๋ ์ด๊ทธ๋ผ์ด๋์์๋ ์คํ๋์ง ์์ต๋๋ค).
tinyvec
ํ์ ๋ฉ๋ชจ๋ฆฌ ํ ๋นํ์ง ์๊ณ ํฌ๊ธฐ ์กฐ์ ์ด ๊ฐ๋ฅํ ์ปจํ
์ด๋(์: Vec
๊ฐ์)๊ฐ ํ์ํ ๋๊ฐ ์์ต๋๋ค. tinyvec
์ ์ฌ์ฉํ๋ฉด ๋ฉ๋๋ค. tinyvec
์์ ๋ฒกํฐ๋ ๋ฐฐ์ด ๋๋ ์ฌ๋ผ์ด์ค๋ก๋ถํฐ ์์ฑ์ด ๋๋ฉฐ, ์ด๋ค์ ์ ์ ์ผ๋ก ํ ๋น๋์๊ฑฐ๋ ์คํ์ ํ ๋น๋์ด ์์ ์ ์์ต๋๋ค.tinyvec
์ ํ์ฌ ๋ฒกํฐ ์์ ์ผ๋ง๋ ๋ง์ ์๋ฆฌ๋จผํธ๋ค์ด ์กด์ฌํ๋ ์ง๋ฅผ ์ถ์ ํ๊ณ ์์ผ๋ฉฐ, ํ ๋น๋ ์๋ณด๋ค ๋ ๋ง์ด ์ฌ์ฉํ๋ ค๊ณ ํ๋ฉด ํจ๋์ ๋ฐ์์ํต๋๋ค.
use tinyvec::{array_vec, ArrayVec}; fn main() { let mut numbers: ArrayVec<[u32; 5]> = array_vec!(42, 66); println!("{numbers:?}"); numbers.push(7); println!("{numbers:?}"); numbers.remove(1); println!("{numbers:?}"); }
tinyvec
๋ฅผ ์ฌ์ฉํ๋ ค๋ฉด ์๋ฆฌ๋จผํธ์ ํ์ ์ดDefault
๋ฅผ ํตํด ์ด๊ธฐํ ๋ ์ ์์ด์ผ ํฉ๋๋ค.- Rust ํ๋ ์ด๊ทธ๋ผ์ด๋์๋
tinyvec
๊ฐ ํฌํจ๋์ด ์์ผ๋ฏ๋ก ์ด ์์๋ ์ธ๋ผ์ธ์ผ๋ก ์คํ๋ฉ๋๋ค.
spin
std::sync::Mutex
๋ฐ std::sync
์ ๊ธฐํ ๋๊ธฐํ ํ๋ฆฌ๋ฏธํฐ๋ธ๋ core
๋๋ alloc
์์ ์ฌ์ฉํ ์ ์์ต๋๋ค. ๊ทธ๋ฌ๋ฉด ์ด๋ป๊ฒ ๋๊ธฐํ ๋๋ interior mutability์ ๊ฐ์ ๊ธฐ๋ฅ์ด ํ์ํ ๊ฒฝ์ฐ ์ด๋ป๊ฒ ํด์ผ ํ ๊น์?
spin
ํฌ๋ ์ดํธ๋ ์ด๋ฌํ ๋๊ธฐํ ํ๋ฆฌ๋ฏธํฐ๋ธ๋ค์ ์คํ๋ก์ผ๋ก ๊ตฌํํ๊ณ ์์ต๋๋ค.
use spin::mutex::SpinMutex; static counter: SpinMutex<u32> = SpinMutex::new(0); fn main() { println!("count: {}", counter.lock()); *counter.lock() += 2; println!("count: {}", counter.lock()); }
- ์ธํฐ๋ฝํธ ํธ๋ค๋ฌ์์ ๋ฝ์ ๊ฑธ ๊ฒฝ์ฐ, ๊ต์ฐฉ ์ํ๊ฐ ๋ฐ์ํ์ง ์๋๋ก ์ฃผ์ํ์ธ์.
spin
์๋ ํฐ์ผ ์ ๊ธ ๋ฎคํ ์ค ๊ตฌํ๋ ์์ต๋๋ค.std::sync
์RwLock
,Barrier
,Once
์ ํด๋นํ๋ ๊ฒ๋ค์ด ์ ๊ณต๋๋ฉฐ, ์ง์ฐ๋ ์ด๊ธฐํ๋ฅผ ์ํLazy
์ ํด๋นํ๋ ๊ฒ๋ ์ ๊ณต๋ฉ๋๋ค.once_cell
ํฌ๋ ์ดํธ์๋ ์ง์ฐ๋ ์ด๊ธฐํ๋ฅผ ์ํ ๋ช ๊ฐ์ง ์ ์ฉํ ํ์ ์ด ์๋๋ฐspin::once::Once
์๋ ์ฝ๊ฐ ๋ค๋ฅธ ์ ๊ทผ ๋ฐฉ์์ ์ฌ์ฉํฉ๋๋ค.- Rust ํ๋ ์ด๊ทธ๋ผ์ด๋์๋
spin
์ด ํฌํจ๋์ด ์์ผ๋ฏ๋ก ์ด ์์๋ ์ธ๋ผ์ธ์ผ๋ก ์คํ๋ฉ๋๋ค.
์๋๋ก์ด๋
AOSP์์ bare-metal Rust ๋ฐ์ด๋๋ฆฌ๋ฅผ ๋น๋ํ๋ ค๋ฉด rust_ffi_static
์ ์ฌ์ฉํ์ฌ Rust ์ฝ๋๋ฅผ ๋น๋ํ๊ณ , ๋ง์ปค ์คํฌ๋ฆฝํธ๊ฐ ํฌํจ๋ cc_binary
๋ฅผ ์ฌ์ฉํ์ฌ ELF ๋ฐ์ด๋๋ฆฌ๋ฅผ ์์ฑํ๊ณ , raw_binary
๋ฅผ ์ฌ์ฉํด ELF๋ฅผ ๊ณง๋ฐ๋ก ์ํ๋ ์ ์๋ ์์(raw) ๋ฐ์ด๋๋ฆฌ๋ก ๋ณํํฉ๋๋ค.
rust_ffi_static {
name: "libvmbase_example",
defaults: ["vmbase_ffi_defaults"],
crate_name: "vmbase_example",
srcs: ["src/main.rs"],
rustlibs: [
"libvmbase",
],
}
cc_binary {
name: "vmbase_example",
defaults: ["vmbase_elf_defaults"],
srcs: [
"idmap.S",
],
static_libs: [
"libvmbase_example",
],
linker_scripts: [
"image.ld",
":vmbase_sections",
],
}
raw_binary {
name: "vmbase_example_bin",
stem: "vmbase_example.bin",
src: ":vmbase_example",
enabled: false,
target: {
android_arm64: {
enabled: true,
},
},
}
vmbase
vmbase ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋, aarch64์ crosvm์์ ์คํ๋๋ VM์ ํ๊ฒํ์ฌ, ์ง์ ์ , UART ์ฝ์ ๋ก๊น , ๋ง์ปค ์คํฌ๋ฆฝํธ, ๋น๋ ๋ฃฐ ๋ฑ์ ๋ํ ๊ธฐ๋ณธ ๊ตฌํ๋ค์ ์ ๊ณตํฉ๋๋ค.
#![no_main] #![no_std] use vmbase::{main, println}; main!(main); pub fn main(arg0: u64, arg1: u64, arg2: u64, arg3: u64) { println!("Hello world"); }
main!
๋งคํฌ๋ก๋vmbase
์ง์ ์ ์์ ํธ์ถ๋ main ํจ์๋ฅผ ํ์ํฉ๋๋ค.vmbase
๊ฐ ์ ๊ณตํ๋ ์ง์ ์ ์ ์ฝ์์ ์ด๊ธฐํ ํ๋ฉฐ, main ํจ์๊ฐ ๋ฆฌํดํ๋ฉด PSCI_SYSTEM_OFF ๋ฉ์์ง๋ฅผ PSCI๋ฅผ ํตํด ๋ณด๋ด์ด์ VM์ ์ข ๋ฃ์ํต๋๋ค.
์ฐ์ต๋ฌธ์
We will write a driver for the PL031 real-time clock device.
After looking at the exercises, you can look at the solutions provided.
RTC driver
The QEMU aarch64 virt machine has a PL031 real-time clock at 0x9010000. For this exercise, you should write a driver for it.
- Use it to print the current time to the serial console. You can use the
chrono
crate for date/time formatting. - Use the match register and raw interrupt status to busy-wait until a given time, e.g. 3 seconds in the future. (Call
core::hint::spin_loop
inside the loop.) - Extension if you have time: Enable and handle the interrupt generated by the RTC match. You can use the driver provided in the
arm-gic
crate to configure the Arm Generic Interrupt Controller.- Use the RTC interrupt, which is wired to the GIC as
IntId::spi(2)
. - Once the interrupt is enabled, you can put the core to sleep via
arm_gic::wfi()
, which will cause the core to sleep until it receives an interrupt.
- Use the RTC interrupt, which is wired to the GIC as
Download the exercise template and look in the rtc
directory for the following files.
src/main.rs
:
#![no_main] #![no_std] mod exceptions; mod logger; mod pl011; use crate::pl011::Uart; use arm_gic::gicv3::GicV3; use core::panic::PanicInfo; use log::{error, info, trace, LevelFilter}; use smccc::psci::system_off; use smccc::Hvc; /// Base addresses of the GICv3. const GICD_BASE_ADDRESS: *mut u64 = 0x800_0000 as _; const GICR_BASE_ADDRESS: *mut u64 = 0x80A_0000 as _; /// Base address of the primary PL011 UART. const PL011_BASE_ADDRESS: *mut u32 = 0x900_0000 as _; #[no_mangle] extern "C" fn main(x0: u64, x1: u64, x2: u64, x3: u64) { // Safe because `PL011_BASE_ADDRESS` is the base address of a PL011 device, // and nothing else accesses that address range. let uart = unsafe { Uart::new(PL011_BASE_ADDRESS) }; logger::init(uart, LevelFilter::Trace).unwrap(); info!("main({:#x}, {:#x}, {:#x}, {:#x})", x0, x1, x2, x3); // Safe because `GICD_BASE_ADDRESS` and `GICR_BASE_ADDRESS` are the base // addresses of a GICv3 distributor and redistributor respectively, and // nothing else accesses those address ranges. let mut gic = unsafe { GicV3::new(GICD_BASE_ADDRESS, GICR_BASE_ADDRESS) }; gic.setup(); // TODO: Create instance of RTC driver and print current time. // TODO: Wait for 3 seconds. system_off::<Hvc>().unwrap(); } #[panic_handler] fn panic(info: &PanicInfo) -> ! { error!("{info}"); system_off::<Hvc>().unwrap(); loop {} }
src/exceptions.rs
(you should only need to change this for the 3rd part of the exercise):
#![allow(unused)] fn main() { // Copyright 2023 Google LLC // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. use arm_gic::gicv3::GicV3; use log::{error, info, trace}; use smccc::psci::system_off; use smccc::Hvc; #[no_mangle] extern "C" fn sync_exception_current(_elr: u64, _spsr: u64) { error!("sync_exception_current"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn irq_current(_elr: u64, _spsr: u64) { trace!("irq_current"); let intid = GicV3::get_and_acknowledge_interrupt().expect("No pending interrupt"); info!("IRQ {intid:?}"); } #[no_mangle] extern "C" fn fiq_current(_elr: u64, _spsr: u64) { error!("fiq_current"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn serr_current(_elr: u64, _spsr: u64) { error!("serr_current"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn sync_lower(_elr: u64, _spsr: u64) { error!("sync_lower"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn irq_lower(_elr: u64, _spsr: u64) { error!("irq_lower"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn fiq_lower(_elr: u64, _spsr: u64) { error!("fiq_lower"); system_off::<Hvc>().unwrap(); } #[no_mangle] extern "C" fn serr_lower(_elr: u64, _spsr: u64) { error!("serr_lower"); system_off::<Hvc>().unwrap(); } }
src/logger.rs
(you shouldnโt need to change this):
#![allow(unused)] fn main() { // Copyright 2023 Google LLC // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // ANCHOR: main use crate::pl011::Uart; use core::fmt::Write; use log::{LevelFilter, Log, Metadata, Record, SetLoggerError}; use spin::mutex::SpinMutex; static LOGGER: Logger = Logger { uart: SpinMutex::new(None), }; struct Logger { uart: SpinMutex<Option<Uart>>, } impl Log for Logger { fn enabled(&self, _metadata: &Metadata) -> bool { true } fn log(&self, record: &Record) { writeln!( self.uart.lock().as_mut().unwrap(), "[{}] {}", record.level(), record.args() ) .unwrap(); } fn flush(&self) {} } /// Initialises UART logger. pub fn init(uart: Uart, max_level: LevelFilter) -> Result<(), SetLoggerError> { LOGGER.uart.lock().replace(uart); log::set_logger(&LOGGER)?; log::set_max_level(max_level); Ok(()) } }
src/pl011.rs
(you shouldnโt need to change this):
#![allow(unused)] fn main() { // Copyright 2023 Google LLC // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #![allow(unused)] use core::fmt::{self, Write}; use core::ptr::{addr_of, addr_of_mut}; // ANCHOR: Flags use bitflags::bitflags; bitflags! { /// Flags from the UART flag register. #[repr(transparent)] #[derive(Copy, Clone, Debug, Eq, PartialEq)] struct Flags: u16 { /// Clear to send. const CTS = 1 << 0; /// Data set ready. const DSR = 1 << 1; /// Data carrier detect. const DCD = 1 << 2; /// UART busy transmitting data. const BUSY = 1 << 3; /// Receive FIFO is empty. const RXFE = 1 << 4; /// Transmit FIFO is full. const TXFF = 1 << 5; /// Receive FIFO is full. const RXFF = 1 << 6; /// Transmit FIFO is empty. const TXFE = 1 << 7; /// Ring indicator. const RI = 1 << 8; } } // ANCHOR_END: Flags bitflags! { /// Flags from the UART Receive Status Register / Error Clear Register. #[repr(transparent)] #[derive(Copy, Clone, Debug, Eq, PartialEq)] struct ReceiveStatus: u16 { /// Framing error. const FE = 1 << 0; /// Parity error. const PE = 1 << 1; /// Break error. const BE = 1 << 2; /// Overrun error. const OE = 1 << 3; } } // ANCHOR: Registers #[repr(C, align(4))] struct Registers { dr: u16, _reserved0: [u8; 2], rsr: ReceiveStatus, _reserved1: [u8; 19], fr: Flags, _reserved2: [u8; 6], ilpr: u8, _reserved3: [u8; 3], ibrd: u16, _reserved4: [u8; 2], fbrd: u8, _reserved5: [u8; 3], lcr_h: u8, _reserved6: [u8; 3], cr: u16, _reserved7: [u8; 3], ifls: u8, _reserved8: [u8; 3], imsc: u16, _reserved9: [u8; 2], ris: u16, _reserved10: [u8; 2], mis: u16, _reserved11: [u8; 2], icr: u16, _reserved12: [u8; 2], dmacr: u8, _reserved13: [u8; 3], } // ANCHOR_END: Registers // ANCHOR: Uart /// Driver for a PL011 UART. #[derive(Debug)] pub struct Uart { registers: *mut Registers, } impl Uart { /// Constructs a new instance of the UART driver for a PL011 device at the /// given base address. /// /// # Safety /// /// The given base address must point to the MMIO control registers of a /// PL011 device, which must be mapped into the address space of the process /// as device memory and not have any other aliases. pub unsafe fn new(base_address: *mut u32) -> Self { Self { registers: base_address as *mut Registers, } } /// Writes a single byte to the UART. pub fn write_byte(&self, byte: u8) { // Wait until there is room in the TX buffer. while self.read_flag_register().contains(Flags::TXFF) {} // Safe because we know that self.registers points to the control // registers of a PL011 device which is appropriately mapped. unsafe { // Write to the TX buffer. addr_of_mut!((*self.registers).dr).write_volatile(byte.into()); } // Wait until the UART is no longer busy. while self.read_flag_register().contains(Flags::BUSY) {} } /// Reads and returns a pending byte, or `None` if nothing has been received. pub fn read_byte(&self) -> Option<u8> { if self.read_flag_register().contains(Flags::RXFE) { None } else { let data = unsafe { addr_of!((*self.registers).dr).read_volatile() }; // TODO: Check for error conditions in bits 8-11. Some(data as u8) } } fn read_flag_register(&self) -> Flags { // Safe because we know that self.registers points to the control // registers of a PL011 device which is appropriately mapped. unsafe { addr_of!((*self.registers).fr).read_volatile() } } } // ANCHOR_END: Uart impl Write for Uart { fn write_str(&mut self, s: &str) -> fmt::Result { for c in s.as_bytes() { self.write_byte(*c); } Ok(()) } } // Safe because it just contains a pointer to device memory, which can be // accessed from any context. unsafe impl Send for Uart {} }
Cargo.toml
(you shouldnโt need to change this):
[workspace]
[package]
name = "rtc"
version = "0.1.0"
edition = "2021"
publish = false
[dependencies]
arm-gic = "0.1.0"
bitflags = "2.0.0"
chrono = { version = "0.4.24", default-features = false }
log = "0.4.17"
smccc = "0.1.1"
spin = "0.9.8"
[build-dependencies]
cc = "1.0.73"
build.rs
(you shouldnโt need to change this):
// Copyright 2023 Google LLC // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. use cc::Build; use std::env; fn main() { #[cfg(target_os = "linux")] env::set_var("CROSS_COMPILE", "aarch64-linux-gnu"); #[cfg(not(target_os = "linux"))] env::set_var("CROSS_COMPILE", "aarch64-none-elf"); Build::new() .file("entry.S") .file("exceptions.S") .file("idmap.S") .compile("empty") }
entry.S
(you shouldnโt need to change this):
/*
* Copyright 2023 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
.macro adr_l, reg:req, sym:req
adrp \reg, \sym
add \reg, \reg, :lo12:\sym
.endm
.macro mov_i, reg:req, imm:req
movz \reg, :abs_g3:\imm
movk \reg, :abs_g2_nc:\imm
movk \reg, :abs_g1_nc:\imm
movk \reg, :abs_g0_nc:\imm
.endm
.set .L_MAIR_DEV_nGnRE, 0x04
.set .L_MAIR_MEM_WBWA, 0xff
.set .Lmairval, .L_MAIR_DEV_nGnRE | (.L_MAIR_MEM_WBWA << 8)
/* 4 KiB granule size for TTBR0_EL1. */
.set .L_TCR_TG0_4KB, 0x0 << 14
/* 4 KiB granule size for TTBR1_EL1. */
.set .L_TCR_TG1_4KB, 0x2 << 30
/* Disable translation table walk for TTBR1_EL1, generating a translation fault instead. */
.set .L_TCR_EPD1, 0x1 << 23
/* Translation table walks for TTBR0_EL1 are inner sharable. */
.set .L_TCR_SH_INNER, 0x3 << 12
/*
* Translation table walks for TTBR0_EL1 are outer write-back read-allocate write-allocate
* cacheable.
*/
.set .L_TCR_RGN_OWB, 0x1 << 10
/*
* Translation table walks for TTBR0_EL1 are inner write-back read-allocate write-allocate
* cacheable.
*/
.set .L_TCR_RGN_IWB, 0x1 << 8
/* Size offset for TTBR0_EL1 is 2**39 bytes (512 GiB). */
.set .L_TCR_T0SZ_512, 64 - 39
.set .Ltcrval, .L_TCR_TG0_4KB | .L_TCR_TG1_4KB | .L_TCR_EPD1 | .L_TCR_RGN_OWB
.set .Ltcrval, .Ltcrval | .L_TCR_RGN_IWB | .L_TCR_SH_INNER | .L_TCR_T0SZ_512
/* Stage 1 instruction access cacheability is unaffected. */
.set .L_SCTLR_ELx_I, 0x1 << 12
/* SP alignment fault if SP is not aligned to a 16 byte boundary. */
.set .L_SCTLR_ELx_SA, 0x1 << 3
/* Stage 1 data access cacheability is unaffected. */
.set .L_SCTLR_ELx_C, 0x1 << 2
/* EL0 and EL1 stage 1 MMU enabled. */
.set .L_SCTLR_ELx_M, 0x1 << 0
/* Privileged Access Never is unchanged on taking an exception to EL1. */
.set .L_SCTLR_EL1_SPAN, 0x1 << 23
/* SETEND instruction disabled at EL0 in aarch32 mode. */
.set .L_SCTLR_EL1_SED, 0x1 << 8
/* Various IT instructions are disabled at EL0 in aarch32 mode. */
.set .L_SCTLR_EL1_ITD, 0x1 << 7
.set .L_SCTLR_EL1_RES1, (0x1 << 11) | (0x1 << 20) | (0x1 << 22) | (0x1 << 28) | (0x1 << 29)
.set .Lsctlrval, .L_SCTLR_ELx_M | .L_SCTLR_ELx_C | .L_SCTLR_ELx_SA | .L_SCTLR_EL1_ITD | .L_SCTLR_EL1_SED
.set .Lsctlrval, .Lsctlrval | .L_SCTLR_ELx_I | .L_SCTLR_EL1_SPAN | .L_SCTLR_EL1_RES1
/**
* This is a generic entry point for an image. It carries out the operations required to prepare the
* loaded image to be run. Specifically, it zeroes the bss section using registers x25 and above,
* prepares the stack, enables floating point, and sets up the exception vector. It preserves x0-x3
* for the Rust entry point, as these may contain boot parameters.
*/
.section .init.entry, "ax"
.global entry
entry:
/* Load and apply the memory management configuration, ready to enable MMU and caches. */
adrp x30, idmap
msr ttbr0_el1, x30
mov_i x30, .Lmairval
msr mair_el1, x30
mov_i x30, .Ltcrval
/* Copy the supported PA range into TCR_EL1.IPS. */
mrs x29, id_aa64mmfr0_el1
bfi x30, x29, #32, #4
msr tcr_el1, x30
mov_i x30, .Lsctlrval
/*
* Ensure everything before this point has completed, then invalidate any potentially stale
* local TLB entries before they start being used.
*/
isb
tlbi vmalle1
ic iallu
dsb nsh
isb
/*
* Configure sctlr_el1 to enable MMU and cache and don't proceed until this has completed.
*/
msr sctlr_el1, x30
isb
/* Disable trapping floating point access in EL1. */
mrs x30, cpacr_el1
orr x30, x30, #(0x3 << 20)
msr cpacr_el1, x30
isb
/* Zero out the bss section. */
adr_l x29, bss_begin
adr_l x30, bss_end
0: cmp x29, x30
b.hs 1f
stp xzr, xzr, [x29], #16
b 0b
1: /* Prepare the stack. */
adr_l x30, boot_stack_end
mov sp, x30
/* Set up exception vector. */
adr x30, vector_table_el1
msr vbar_el1, x30
/* Call into Rust code. */
bl main
/* Loop forever waiting for interrupts. */
2: wfi
b 2b
exceptions.S
(you shouldnโt need to change this):
/*
* Copyright 2023 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* Saves the volatile registers onto the stack. This currently takes 14
* instructions, so it can be used in exception handlers with 18 instructions
* left.
*
* On return, x0 and x1 are initialised to elr_el2 and spsr_el2 respectively,
* which can be used as the first and second arguments of a subsequent call.
*/
.macro save_volatile_to_stack
/* Reserve stack space and save registers x0-x18, x29 & x30. */
stp x0, x1, [sp, #-(8 * 24)]!
stp x2, x3, [sp, #8 * 2]
stp x4, x5, [sp, #8 * 4]
stp x6, x7, [sp, #8 * 6]
stp x8, x9, [sp, #8 * 8]
stp x10, x11, [sp, #8 * 10]
stp x12, x13, [sp, #8 * 12]
stp x14, x15, [sp, #8 * 14]
stp x16, x17, [sp, #8 * 16]
str x18, [sp, #8 * 18]
stp x29, x30, [sp, #8 * 20]
/*
* Save elr_el1 & spsr_el1. This such that we can take nested exception
* and still be able to unwind.
*/
mrs x0, elr_el1
mrs x1, spsr_el1
stp x0, x1, [sp, #8 * 22]
.endm
/**
* Restores the volatile registers from the stack. This currently takes 14
* instructions, so it can be used in exception handlers while still leaving 18
* instructions left; if paired with save_volatile_to_stack, there are 4
* instructions to spare.
*/
.macro restore_volatile_from_stack
/* Restore registers x2-x18, x29 & x30. */
ldp x2, x3, [sp, #8 * 2]
ldp x4, x5, [sp, #8 * 4]
ldp x6, x7, [sp, #8 * 6]
ldp x8, x9, [sp, #8 * 8]
ldp x10, x11, [sp, #8 * 10]
ldp x12, x13, [sp, #8 * 12]
ldp x14, x15, [sp, #8 * 14]
ldp x16, x17, [sp, #8 * 16]
ldr x18, [sp, #8 * 18]
ldp x29, x30, [sp, #8 * 20]
/* Restore registers elr_el1 & spsr_el1, using x0 & x1 as scratch. */
ldp x0, x1, [sp, #8 * 22]
msr elr_el1, x0
msr spsr_el1, x1
/* Restore x0 & x1, and release stack space. */
ldp x0, x1, [sp], #8 * 24
.endm
/**
* This is a generic handler for exceptions taken at the current EL while using
* SP0. It behaves similarly to the SPx case by first switching to SPx, doing
* the work, then switching back to SP0 before returning.
*
* Switching to SPx and calling the Rust handler takes 16 instructions. To
* restore and return we need an additional 16 instructions, so we can implement
* the whole handler within the allotted 32 instructions.
*/
.macro current_exception_sp0 handler:req
msr spsel, #1
save_volatile_to_stack
bl \handler
restore_volatile_from_stack
msr spsel, #0
eret
.endm
/**
* This is a generic handler for exceptions taken at the current EL while using
* SPx. It saves volatile registers, calls the Rust handler, restores volatile
* registers, then returns.
*
* This also works for exceptions taken from EL0, if we don't care about
* non-volatile registers.
*
* Saving state and jumping to the Rust handler takes 15 instructions, and
* restoring and returning also takes 15 instructions, so we can fit the whole
* handler in 30 instructions, under the limit of 32.
*/
.macro current_exception_spx handler:req
save_volatile_to_stack
bl \handler
restore_volatile_from_stack
eret
.endm
.section .text.vector_table_el1, "ax"
.global vector_table_el1
.balign 0x800
vector_table_el1:
sync_cur_sp0:
current_exception_sp0 sync_exception_current
.balign 0x80
irq_cur_sp0:
current_exception_sp0 irq_current
.balign 0x80
fiq_cur_sp0:
current_exception_sp0 fiq_current
.balign 0x80
serr_cur_sp0:
current_exception_sp0 serr_current
.balign 0x80
sync_cur_spx:
current_exception_spx sync_exception_current
.balign 0x80
irq_cur_spx:
current_exception_spx irq_current
.balign 0x80
fiq_cur_spx:
current_exception_spx fiq_current
.balign 0x80
serr_cur_spx:
current_exception_spx serr_current
.balign 0x80
sync_lower_64:
current_exception_spx sync_lower
.balign 0x80
irq_lower_64:
current_exception_spx irq_lower
.balign 0x80
fiq_lower_64:
current_exception_spx fiq_lower
.balign 0x80
serr_lower_64:
current_exception_spx serr_lower
.balign 0x80
sync_lower_32:
current_exception_spx sync_lower
.balign 0x80
irq_lower_32:
current_exception_spx irq_lower
.balign 0x80
fiq_lower_32:
current_exception_spx fiq_lower
.balign 0x80
serr_lower_32:
current_exception_spx serr_lower
idmap.S
(you shouldnโt need to change this):
/*
* Copyright 2023 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
.set .L_TT_TYPE_BLOCK, 0x1
.set .L_TT_TYPE_PAGE, 0x3
.set .L_TT_TYPE_TABLE, 0x3
/* Access flag. */
.set .L_TT_AF, 0x1 << 10
/* Not global. */
.set .L_TT_NG, 0x1 << 11
.set .L_TT_XN, 0x3 << 53
.set .L_TT_MT_DEV, 0x0 << 2 // MAIR #0 (DEV_nGnRE)
.set .L_TT_MT_MEM, (0x1 << 2) | (0x3 << 8) // MAIR #1 (MEM_WBWA), inner shareable
.set .L_BLOCK_DEV, .L_TT_TYPE_BLOCK | .L_TT_MT_DEV | .L_TT_AF | .L_TT_XN
.set .L_BLOCK_MEM, .L_TT_TYPE_BLOCK | .L_TT_MT_MEM | .L_TT_AF | .L_TT_NG
.section ".rodata.idmap", "a", %progbits
.global idmap
.align 12
idmap:
/* level 1 */
.quad .L_BLOCK_DEV | 0x0 // 1 GiB of device mappings
.quad .L_BLOCK_MEM | 0x40000000 // 1 GiB of DRAM
.fill 254, 8, 0x0 // 254 GiB of unmapped VA space
.quad .L_BLOCK_DEV | 0x4000000000 // 1 GiB of device mappings
.fill 255, 8, 0x0 // 255 GiB of remaining VA space
image.ld
(you shouldnโt need to change this):
/*
* Copyright 2023 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/*
* Code will start running at this symbol which is placed at the start of the
* image.
*/
ENTRY(entry)
MEMORY
{
image : ORIGIN = 0x40080000, LENGTH = 2M
}
SECTIONS
{
/*
* Collect together the code.
*/
.init : ALIGN(4096) {
text_begin = .;
*(.init.entry)
*(.init.*)
} >image
.text : {
*(.text.*)
} >image
text_end = .;
/*
* Collect together read-only data.
*/
.rodata : ALIGN(4096) {
rodata_begin = .;
*(.rodata.*)
} >image
.got : {
*(.got)
} >image
rodata_end = .;
/*
* Collect together the read-write data including .bss at the end which
* will be zero'd by the entry code.
*/
.data : ALIGN(4096) {
data_begin = .;
*(.data.*)
/*
* The entry point code assumes that .data is a multiple of 32
* bytes long.
*/
. = ALIGN(32);
data_end = .;
} >image
/* Everything beyond this point will not be included in the binary. */
bin_end = .;
/* The entry point code assumes that .bss is 16-byte aligned. */
.bss : ALIGN(16) {
bss_begin = .;
*(.bss.*)
*(COMMON)
. = ALIGN(16);
bss_end = .;
} >image
.stack (NOLOAD) : ALIGN(4096) {
boot_stack_begin = .;
. += 40 * 4096;
. = ALIGN(4096);
boot_stack_end = .;
} >image
. = ALIGN(4K);
PROVIDE(dma_region = .);
/*
* Remove unused sections from the image.
*/
/DISCARD/ : {
/* The image loads itself so doesn't need these sections. */
*(.gnu.hash)
*(.hash)
*(.interp)
*(.eh_frame_hdr)
*(.eh_frame)
*(.note.gnu.build-id)
}
}
Makefile
(you shouldnโt need to change this):
# Copyright 2023 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
UNAME := $(shell uname -s)
ifeq ($(UNAME),Linux)
TARGET = aarch64-linux-gnu
else
TARGET = aarch64-none-elf
endif
OBJCOPY = $(TARGET)-objcopy
.PHONY: build qemu_minimal qemu qemu_logger
all: rtc.bin
build:
cargo build
rtc.bin: build
$(OBJCOPY) -O binary target/aarch64-unknown-none/debug/rtc $@
qemu: rtc.bin
qemu-system-aarch64 -machine virt,gic-version=3 -cpu max -serial mon:stdio -display none -kernel $< -s
clean:
cargo clean
rm -f *.bin
.cargo/config.toml
(you shouldnโt need to change this):
[build]
target = "aarch64-unknown-none"
rustflags = ["-C", "link-arg=-Timage.ld"]
Run the code in QEMU with make qemu
.
Welcome to Concurrency in Rust
๋ฌ์คํธ๋ ๋์์ฑ ์ง์์ด ๋ง๊ฐํฉ๋๋ค. ์ด์์ฒด์ ๋ ๋ฒจ์ ์ค๋ ๋๋ฅผ ์ฌ์ฉํ๋ฉฐ, ๋ฎคํ์ค์ ์ฑ๋๋ ์ง์ํฉ๋๋ค.
๋ฌ์คํธ์ ํ์ ์์คํ ์ ํ๋ก๊ทธ๋จ์ ๋์์ฑ ๋ฒ๊ทธ๊ฐ ์์ ๊ฒฝ์ฐ, ์ปดํ์ผ ์์ ์๋ฌ๊ฐ ๋ฐ์ํ๋๋ก ํด ์ค๋๋ค. ์ปดํ์ผ๋ฌ๋ฅผ ์ด์ฉํด์ ํ๋ก๊ทธ๋จ์ด ์ํ์์ ์ ํํ ๋์ํจ์ ๋ฏธ๋ฆฌ ๋ณด์ฅํด ์ฃผ๊ธฐ ๋๋ฌธ์, ์ฌ๋๋ค์ ์ด๋ฅผ ์ข ์ข ๊ฒ์๋ ๋์์ฑ ์ด๋ผ๊ณ ํฉ๋๋ค.
์ค๋ ๋
๋ฌ์คํธ์ ์ค๋ ๋๋ ๋ค๋ฅธ ์ธ์ด์ ์ค๋ ๋์ ์ ์ฌํ๊ฒ ๋์ํฉ๋๋ค:
use std::thread; use std::time::Duration; fn main() { thread::spawn(|| { for i in 1..10 { println!("Count in thread: {i}!"); thread::sleep(Duration::from_millis(5)); } }); for i in 1..5 { println!("Main thread: {i}"); thread::sleep(Duration::from_millis(5)); } }
- ์ค๋ ๋๋ ๋ชจ๋ ๋ฐ๋ชฌ ์ค๋ ๋์ ๋๋ค. ๋ฐ๋ผ์ ๋ฉ์ธ ์ค๋ ๋๋ ์ด ์ค๋ ๋๋ค์ด ๋๋๊ธฐ๋ฅผ ๊ธฐ๋ค๋ฆฌ์ง ์์ต๋๋ค.
- ํ ์ค๋ ๋์์ ๋ฐ์ํ ํจ๋์ ๋ค๋ฅธ ์ค๋ ๋์๊ฒ ์ํฅ์ ๋ผ์น์ง ์์ต๋๋ค.
- ํจ๋์ ์ถ๊ฐ์ ๋ณด(ํ์ด๋ก๋)๋ฅผ ํฌํจํ ์ ์์ผ๋ฉฐ, ์ด๋
downcast_ref
๋ก ํ์ด๋ณผ ์ ์์ต๋๋ค.
- ํจ๋์ ์ถ๊ฐ์ ๋ณด(ํ์ด๋ก๋)๋ฅผ ํฌํจํ ์ ์์ผ๋ฉฐ, ์ด๋
ํค ํฌ์ธํธ:
-
๋ฉ์ธ ์ค๋ ๋๊ฐ ์์ ์ค๋ ๋๋ฅผ ๊ธฐ๋ค๋ฆฌ์ง ์๊ธฐ ๋๋ฌธ์ ์์ ์ค๋ ๋์ for๋ฌธ์ 10๊น์ง ๊ฐ์ง ์์ต๋๋ค.
-
๋ง์ฝ ๋ฉ์ธ ์ค๋ ๋๊ฐ ์์ ์ค๋ ๋๊ฐ ๋๋ ๋ ๊น์ง ๊ธฐ๋ค๋ฆฌ๊ธฐ๋ฅผ ์ํ๋ค๋ฉด
let handle = thread::spawn(...)
์ผ๋ก ์ค๋ ๋๋ฅผ ์ ์ธํ ํhandle.join()
๋ก ์ฐ๊ฒฐํ์ฌ ์ฌ์ฉํฉ๋๋ค. -
์์ ์ค๋ ๋์์ ๋ฐ์ํ ํจ๋์ด ๋ฉ์ธ ์ค๋ ๋์๋ ์ํฅ์ ์ฃผ์ง ์์์ ํ์ธํ์๊ธฐ ๋ฐ๋๋๋ค.
-
handle.join()
์ฌ์ฉ์Result
๋ฐํ๊ฐ์ ํตํด ํจ๋์ ์ถ๊ฐ์ ๋ณด์ ์ ๊ทผํ ์ ์์ต๋๋ค. ์ด ์์ ์์Any
์ ๋ํด ์ด์ผ๊ธฐ๋ฅผ ํด ๋ณด๋ฉด ์ข์ต๋๋ค.
๋ฒ์ ์ค๋ ๋(Scoped Threads)
๋ณดํต, ์ค๋ ๋๋ ์ค๋ ๋ ๋ฐ์์ ๋ฐ์ดํฐ๋ฅผ ๋น๋ฆด ์ ์์ต๋๋ค:
use std::thread; fn foo() { let s = String::from("Hello"); thread::spawn(|| { println!("Length: {}", s.len()); }); } fn main() { foo(); }
ํ์ง๋ง, scoped thread์์๋ ๊ฐ๋ฅํฉ๋๋ค:
use std::thread; fn main() { let s = String::from("Hello"); thread::scope(|scope| { scope.spawn(|| { println!("Length: {}", s.len()); }); }); }
thread::scope
ํจ์๊ฐ ์๋ฃ๋๋ฉด ๊ทธ ์์์ ์์ฑ๋ ๋ชจ๋ ์ค๋ ๋๋ค์ด ์ข ๋ฃํ์์ด ๋ณด์ฅ๋๊ธฐ ๋๋ฌธ์, ๊ทธ ๋ ๋น๋ ธ๋ ๋ฐ์ดํฐ๋ค์ ๋ค์ ๋ฐํํ ์ ์๊ธฐ ๋๋ฌธ์ ๋๋ค.- ์ผ๋ฐ์ ์ธ ๋ฌ์คํธ์ ๋น๋ฆผ ๊ท์น์ด ์ ์ฉ๋ฉ๋๋ค: ํ ์ค๋ ๋์ ์ํ ๊ฐ๋ณ ๋น๋ฆผ ๋๋ ์ฌ๋ฌ ์ค๋ ๋์ ๋ํ ๋ถ๋ณ ๋น๋ฆผ์ค ํ๋๋ง ๊ฐ๋ฅํฉ๋๋ค.
์ฑ๋
๋ฌ์คํธ์ ์ฑ๋์ Sender<T>
์ Receiver<T>
๋ ๋ถ๋ถ์ผ๋ก ๊ตฌ์ฑ๋ฉ๋๋ค. ์ด ๋์ ์ฑ๋์ ํตํด ์๋ก ์ฐ๊ฒฐ๋์ด ์์ง๋ง, ์ฐ๋ฆฌ๋ ์ฑ๋์ ๋ณผ ์๋ ์๊ณ ์ด ์ ๋๋จ๋ง์ ์ฌ์ฉํ๊ฒ ๋ฉ๋๋ค.
use std::sync::mpsc; use std::thread; fn main() { let (tx, rx) = mpsc::channel(); tx.send(10).unwrap(); tx.send(20).unwrap(); println!("Received: {:?}", rx.recv()); println!("Received: {:?}", rx.recv()); let tx2 = tx.clone(); tx2.send(30).unwrap(); println!("Received: {:?}", rx.recv()); }
mpsc
๋ โMulti-Produce, Single-Consumerโ๋ฅผ ์๋ฏธํฉ๋๋ค.Sender
์SyncSender
๋Clone
์ ๊ตฌํํ์ง๋ง (์ฆ, ์ฌ๋ฌ๊ฐ์ producer๋ฅผ ๋ง๋ค์ ์์ต๋๋ค)Receiver
๋Clone
์ ๊ตฌํํ์ง ์์ต๋๋ค.send()
์recv()
๋Result
๋ฅผ ๋ฐํํฉ๋๋ค. ๋ง์ผErr
๊ฐ ๋ฐํ๋๋ค๋ฉด, ์๋๋ฐฉ์Sender
๋๋Receiver
๊ฐ ์ญ์ ๋์๊ณ ์ฑ๋์ด ๋ซํ๋ค๋ ๋ป์ ๋๋ค.
๋ฌด๊ฒฝ๊ณ ์ฑ๋
mpsc::channel()
ํจ์๋ ๊ฒฝ๊ณ๊ฐ ์๋ ๋น๋๊ธฐ ์ฑ๋์ ์์ฑํฉ๋๋ค:
use std::sync::mpsc; use std::thread; use std::time::Duration; fn main() { let (tx, rx) = mpsc::channel(); thread::spawn(move || { let thread_id = thread::current().id(); for i in 1..10 { tx.send(format!("Message {i}")).unwrap(); println!("{thread_id:?}: sent Message {i}"); } println!("{thread_id:?}: done"); }); thread::sleep(Duration::from_millis(100)); for msg in rx.iter() { println!("Main: got {msg}"); } }
๊ฒฝ๊ณ ์ฑ๋
With bounded (synchronous) channels, send
can block the current thread:
use std::sync::mpsc; use std::thread; use std::time::Duration; fn main() { let (tx, rx) = mpsc::sync_channel(3); thread::spawn(move || { let thread_id = thread::current().id(); for i in 1..10 { tx.send(format!("Message {i}")).unwrap(); println!("{thread_id:?}: sent Message {i}"); } println!("{thread_id:?}: done"); }); thread::sleep(Duration::from_millis(100)); for msg in rx.iter() { println!("Main: got {msg}"); } }
- Calling
send
will block the current thread until there is space in the channel for the new message. The thread can be blocked indefinitely if there is nobody who reads from the channel. - A call to
send
will abort with an error (that is why it returnsResult
) if the channel is closed. A channel is closed when the receiver is dropped. - A bounded channel with a size of zero is called a โrendezvous channelโ. Every send will block the current thread until another thread calls
read
.
Send
์ Sync
๋ฌ์คํธ๋ ์ด๋ป๊ฒ ํด์ ์ค๋ ๋ ๊ฐ์ ํน์ ๋ฐ์ดํฐ ํ์ ์ด ๊ณต์ ๋ ์ ์๊ฑฐ๋, ์๋๋ค๋ ๊ฒ์ ์๊น์? ์ ๋ต์ ๋ค์ ๋ ํธ๋ ์์ ์์ต๋๋ค:
Send
:T
๊ฐ ์ค๋ ๋ ๊ฐ ์ด๋์ด ์์ ํ๋ค๋ฉด,T
์ ํ์ ์Send
์ ๋๋ค.Sync
:&T
๊ฐ ์ค๋ ๋ ๊ฐ ์ด๋์ด ์์ ํ๋ค๋ฉด,&T
์ ํ์ ์Sync
์ ๋๋ค.
Send
์ Sync
ํธ๋ ์์ ์์ ํ์ง ์์ ํธ๋ ์์
๋๋ค. ์ปดํ์ผ๋ฌ๋ ํ์
์ ์์๋ค์ด ๋ชจ๋ Send
์ Sync
ํ์
์ด๋ฉด ์๋์ผ๋ก ์ด ํธ๋ ์๋ค์ ์ ์ฉ์์ผ ์ค๋๋ค. ๋ฌผ๋ก ์ฌ๋ฌ๋ถ ์ค์ค๋ก ๋ง๋ค๊ณ ์๊ณ ์๋ค๋ฉด ์ง์ ๊ตฌํํด๋ ๋ฉ๋๋ค.
Sync
์Send
๋ ์ด๋ค ํ์ ์ด ํน์ ํ ์ค๋ ๋-์์ ์์ฑ์ ๊ฐ์ง์ ๋ํ๋ด๋ ๋ง์ปค๋ก ์๊ฐํ ์ ์์ต๋๋ค.- ์ด ๋ ํธ๋ ์ดํธ๋ ์ ๋๋ฆญ์์ ์ ์ฝ ์กฐ๊ฑด์ ๋ํ๋ด๋ ํธ๋ ์ดํธ๋ก ์ฌ์ฉ๋ ์๋ ์์ต๋๋ค.
Send
T
๊ฐ ์ค๋ ๋ ๊ฐ์ ์์ ํ๊ฒ ์ด๋๋ ์ ์๋ค๋ฉด,T
์ ํ์ ์Send
์ ๋๋ค.
์์ ๊ถ์ ๋ค๋ฅธ ์ค๋ ๋๋ก ์ด๋ํ๋ฉด ์๋ฉธ์๊ฐ ํด๋น ์ค๋ ๋์์ ์คํ๋ฉ๋๋ค. ์ฌ๊ธฐ์ ์๋ฌธ์ โ์ธ์ ํ ์ค๋ ๋์์ ๊ฐ์ ํ ๋นํ๊ณ ๋ค๋ฅธ ์ค๋ ๋์์ ๊ฐ์ ํ ๋น ํด์ ํ ์ ์๋๊ฐโ ์ ๋๋ค.
์๋ฅผ ๋ค์ด SQLite ๋ผ์ด๋ธ๋ฌ๋ฆฌ ์ฐ๊ฒฐ์ ๋จ์ผ ์ค๋ ๋์์๋ง ์ก์ธ์คํด์ผ ํฉ๋๋ค.
Sync
&T
๊ฐ ์ฌ๋ฌ ์ค๋ ๋์์ ์์ ํ๊ฒ ์ ๊ทผ๋ ์ ์๋ค๋ฉด,&T
์ ํ์ ์Sync
์ ๋๋ค.
์ข ๋ ์ ํํ ์ ์๋ ๋ค์๊ณผ ๊ฐ์ต๋๋ค:
&T
๊ฐSend
์ธ ๊ฒฝ์ฐ์๋งT
์ ํ์ ์ดSync
๊ฐ ๋ฉ๋๋ค
์ ๋ฌธ์ฅ์ ํ์ด์ ์ด์ผ๊ธฐ ํ๋ฉด, ์ด๋ค ํ์ ์ด ์ค๋ ๋ ๊ฐ์ ๊ณต์ ๋์ด์ ์ฌ์ฉ๋๊ธฐ์ ์์ ํ๋ค๋ฉด ๊ทธ ํ์ ์ ์ฐธ์กฐ ํ์ ์ ์ค๋ ๋ ๊ฐ์ ์ด๋ ๊ฐ๋ฅํ๋ค๋ ๊ฒ์ ๋๋ค.
์ด๋ ๋ค์๊ณผ ๊ฐ์ด ์ฆ๋ช
ํ ์ ์์ต๋๋ค: ์ด๋ค ํ์
์ด Sync
๋ผ๋ ๋ง์ ๊ณง ๊ทธ ํ์
์ด ์ฌ๋ฌ ์ค๋ ๋๋ค ์ฌ์ด์์ ๋ฐ์ดํฐ ๋ ์ด์ค๋ ์ฌํ ๋๊ธฐํ ๋ฌธ์ ์์ด ๊ณต์ ๊ฐ๋ฅํ๋ค๋ ๋ง์
๋๋ค. ์ค๋ ๋ ๊ฐ ๊ณต์ ๊ฐ ์์ ํ๋ค๋ฉด, ์ค๋ ๋๊ฐ ์ด๋๋ ์์ ํ ์ ๋ฐ์ ์์ต๋๋ค. ์ด๋ค ํ์
์ ์ค๋ ๋๊ฐ ์ด๋์ด ์์ ํ๋ค๋ฉด, ๊ทธ ํ์
์ ์ฐธ์กฐ ๋ํ ์ค๋ ๋๊ฐ ์ด๋์ด ์์ ํ ์ ๋ฐ์ ์์ต๋๋ค.
์์
Send + Sync
์ฌ๋ฌ๋ถ์ด ๋ค๋ฃจ๊ฒ ๋ ๋๋ถ๋ถ์ ํ์
์ Send + Sync
์
๋๋ค:
i8
,f32
,bool
,char
,&str
, โฆ(T1, T2)
,[T; N]
,&[T]
,struct { x: T }
, โฆString
,Option<T>
,Vec<T>
,Box<T>
, โฆArc<T>
: ์ฐธ์กฐ ์นด์ดํธ ์กฐ์์ ์ํ ๋ฏน ํ๊ธฐ ๋๋ฌธ์ ์ค๋ ๋ ์์ ํจ.Mutex<T>
: ๊ฐ์ ์ ๊ทผํ๊ธฐ ์ํด ๋ฎคํ์ค๋ฅผ ์ ๊ถ์ผ ํ๊ธฐ ๋๋ฌธ์ ์ค๋ ๋ ์์ ํจ.AtomicBool
,AtomicU8
, โฆ: ๊ฐ์ ์ ๊ทผํ ๋ ํน๋ณํ ์ํ ๋ฏน ๋ช ๋ น์ด๋ค์ ์ฌ์ฉํฉ๋๋ค.
์ ๋ค๋ฆญ ํ์
์ ์ผ๋ฐ์ ์ผ๋ก ํ์
ํ๋ผ๋ฉํฐ๊ฐ Send + Sync
์ด๋ฉด Send + Sync
์
๋๋ค.
Send + !Sync
์๋ ํ์ ๋ค์ ๋ค๋ฅธ ์ค๋ ๋๋ก ์ด๋๋ ์ ์์ง๋ง ๋ด๋ถ์ ์ผ๋ก ๊ฐ์ด ๋ณ๊ฒฝ๋ ์ ์๊ธฐ ๋๋ฌธ์ ์ค๋ ๋ ์์ ํ์ง ์์ต๋๋ค:
mpsc::Sender<T>
mpsc::Receiver<T>
Cell<T>
RefCell<T>
!Send + Sync
์๋ ํ์ ๋ค์ ์ค๋ ๋ ์์ ํ์ง๋ง ๋ค๋ฅธ ์ค๋ ๋๋ก ์ด๋๋ ์ ์์ต๋๋ค:
MutexGuard<T>
: ๋ด๋ถ์ ์ผ๋ก, ์ด์์ฒด์ ๊ฐ ์ ๊ณตํ๋ primitive๋ฅผ ์ฌ์ฉํ๋๋ฐ, ์ด primitive๋ ์์ฑ๋ ์ค๋ ๋์์ ํด์ ๊ฐ ์ด๋ฃจ์ด์ ธ์ผ ํฉ๋๋ค. (์ญ์ฃผ: ๊ทธ๋์ ๋ค๋ฅธ ์ค๋ ๋๋ก ์ฎ๊ธธ ์ ์์ต๋๋ค.)
!Send + !Sync
์๋ ํ์ ๋ค์ ์ค๋ ๋ ์์ ํ์ง๋ ์๊ณ ๋ค๋ฅธ ์ค๋ ๋๋ก ์ด๋๋ ์๋ ์์ต๋๋ค:
Rc<T>
:Rc<T>
๋ ์ํ ๋ฏนํ์ง ์์ ๋ฐฉ์์ผ๋ก ์ฐธ์กฐ ์นด์ดํธ๋ฅผ ์กฐ์ํ๋RcBox<T>
๋ฅผ ์ฐธ์กฐํฉ๋๋ค.*const T
,*mut T
: ๋ฌ์คํธ๋ ํฌ์ธํฐ๊ฐ ์ค๋ ๋ ์์ ํ์ง ์๋ค๊ณ ๊ฐ์ ํฉ๋๋ค.
์ํ ๊ณต์
๋ฌ์คํธ๋ ์ฃผ๋ก ์๋ ๋ ๊ฐ์ง ํ์ ์ ์ด์ฉํด์ ๊ณต์ ๋ฐ์ดํฐ ๋๊ธฐํ๋ฅผ ์ํํฉ๋๋ค:
Arc<T>
,T
์ ๋ํ ์ํ ๋ฏน ์ฐธ์กฐ ์นด์ดํธ: ์ด ์ฐธ์กฐ๋ ๋ค์์ ์ค๋ ๋ ์ฌ์ด์์ ๊ณต์ ๋ ์ ์๊ณ , ์ฐธ์กฐํ๋ ๋ง์ง๋ง ์ค๋ ๋๊ฐ ์ข ๋ฃํ ๊ฒฝ์ฐT
๋ฅผ ๋ฐํํฉ๋๋ค.Mutex<T>
:T
๊ฐ์ ๋ํ ์ํธ ๋ฐฐ์ ์์ธ์ค๋ฅผ ๋ณด์ฅํฉ๋๋ค.
Arc
Arc<T>
allows shared read-only access via Arc::clone
:
use std::thread; use std::sync::Arc; fn main() { let v = Arc::new(vec![10, 20, 30]); let mut handles = Vec::new(); for _ in 1..5 { let v = Arc::clone(&v); handles.push(thread::spawn(move || { let thread_id = thread::current().id(); println!("{thread_id:?}: {v:?}"); })); } handles.into_iter().for_each(|h| h.join().unwrap()); println!("v: {v:?}"); }
Arc
stands for โAtomic Reference Countedโ, a thread safe version ofRc
that uses atomic operations.Arc<T>
implementsClone
whether or notT
does. It implementsSend
andSync
if and only ifT
implements them both.Arc::clone()
has the cost of atomic operations that get executed, but after that the use of theT
is free.- Beware of reference cycles,
Arc
does not use a garbage collector to detect them.std::sync::Weak
can help.
Mutex
Mutex<T>
๋ฅผ ์ด์ฉํ๋ฉด ๋ถ๋ณ ์ฐธ์กฐ๋ฅผ ํตํด์๋ T
์ ๊ฐ์ ์์ ํ ์๊ฐ ์์ผ๋ฉฐ, ์ด์ ๋ํด์ ํ ๋ฒ์ ํ ์ค๋ ๋๋ง T
์ ๊ฐ์ ์ ๊ทผ(์ฝ๊ฑฐ๋ ์ฐ๊ฑฐ๋)ํจ์ ๋ณด์ฅํด ์ค๋๋ค:
use std::sync::Mutex; fn main() { let v = Mutex::new(vec![10, 20, 30]); println!("v: {:?}", v.lock().unwrap()); { let mut guard = v.lock().unwrap(); guard.push(40); } println!("v: {:?}", v.lock().unwrap()); }
๋ชจ๋ Mutex<T>
๋ impl<T: Send> Sync for Mutex<T>
๋ฅผ ์๋์ผ๋ก ๊ตฌํํจ์ ์ฐธ์กฐํ์ธ์.
Mutex
in Rust looks like a collection with just one element - the protected data.- It is not possible to forget to acquire the mutex before accessing the protected data.
- You can get an
&mut T
from an&Mutex<T>
by taking the lock. TheMutexGuard
ensures that the&mut T
doesnโt outlive the lock being held. Mutex<T>
implements bothSend
andSync
iff (if and only if)T
implementsSend
.- A read-write lock counterpart -
RwLock
. - Why does
lock()
return aResult
?- If the thread that held the
Mutex
panicked, theMutex
becomes โpoisonedโ to signal that the data it protected might be in an inconsistent state. Callinglock()
on a poisoned mutex fails with aPoisonError
. You can callinto_inner()
on the error to recover the data regardless.
- If the thread that held the
์์
Arc
์ Mutex
์ ๋์์ ์ดํด๋ด
์๋ค:
use std::thread; // use std::sync::{Arc, Mutex}; fn main() { let v = vec![10, 20, 30]; let handle = thread::spawn(|| { v.push(10); }); v.push(1000); handle.join().unwrap(); println!("v: {v:?}"); }
๊ฐ๋ฅํ ํด๊ฒฐ์ฑ :
use std::sync::{Arc, Mutex}; use std::thread; fn main() { let v = Arc::new(Mutex::new(vec![10, 20, 30])); let v2 = Arc::clone(&v); let handle = thread::spawn(move || { let mut v2 = v2.lock().unwrap(); v2.push(10); }); { let mut v = v.lock().unwrap(); v.push(1000); } handle.join().unwrap(); println!("v: {v:?}"); }
๋์ฌ๊ฒจ ๋ณผ ๋ถ๋ถ:
v
๋Arc
์Mutex
๋ชจ๋์ ํฌํจ๋์ด ์์ต๋๋ค. ์ด๋Arc
์Mutex
๊ฐ ์๋ก ์์ ํ ๋ค๋ฅธ ๋ฌธ์ ๋ฅผ ์ํ ๋๊ตฌ์ด๊ธฐ ๋๋ฌธ์ ๋๋ค.Mutex
๋ฅผArc
๋ก ๋ํํ๋ ๊ฒ์ ๊ฐ๋ณ ์ํ๋ฅผ ์ค๋ ๋๋ค ๊ฐ์ ๊ณต์ ํ ๋ ํํ ์ฌ์ฉํ๋ ํจํด์ ๋๋ค.
v: Arc<_>
๋ฅผ ๋ค๋ฅธ ์ค๋ ๋์์ ์ฌ์ฉํ๋ ค๋ฉด, ๋จผ์ v2
๋ก ๋ณต์ฌ๋ฅผ ํ๊ณ ์ด๋ฅผ ๊ทธ ์ค๋ ๋๋ก ์ด๋ ํด์ผ ํฉ๋๋ค. ๊ทธ๋์ ๋๋ค์ ์๊ทธ๋์ฒ์move
๊ฐ ์๋ ๊ฒ์ ๋๋ค.- ๋ธ๋ก์
LockGuard
์ ๋ฒ์๋ฅผ ์ต๋ํ ์ขํ๊ธฐ ์ํด ์ฌ์ฉ๋์์ต๋๋ค.
์ฐ์ต๋ฌธ์
๋์์ฑ ๊ธฐ๋ฒ๋ค์ ์ฐ์ตํด ๋ด ์๋ค
-
์์ฌํ๋ ์ฒ ํ์ ๋ฌธ์ : ๊ณ ์ ์ ์ธ ๋์์ฑ ๋ฌธ์ ์ ๋๋ค.
-
๋ฉํฐ ์ค๋ ๋ ๋งํฌ ๊ฒ์ฌ๊ธฐ: ๋ณ๋ ฌ์ ์ผ๋ก ์นํ์ด์ง์ ๋งํฌ๋ค์ ์ฒดํฌํฉ๋๋ค. ์นด๊ณ ๋ฅผ ํตํด ๋ช ๊ฐ์ง ์์กด์ฑ๋ค์ ๋ค์ด๋๋ฅด ๋ฐ์์ผ ํ๋ ํฐ ํ๋ก์ ํธ ์ ๋๋ค.
After looking at the exercises, you can look at the solutions provided.
์์ฌํ๋ ์ฒ ํ์๋ค
์์ฌํ๋ ์ฒ ํ์ ๋ฌธ์ ๋ ๋์์ฑ์ ์์ด์ ๊ณ ์ ์ ์ธ ๋ฌธ์ ์ ๋๋ค:
5๋ช ์ ์ฒ ํ์๊ฐ ์ํ์์ ์์ฌ๋ฅผ ํ๊ณ ์์ต๋๋ค. ์ฒ ํ์๋ ์ํ์์ ์์ ์ ์๋ฆฌ์ ์์์์ต๋๋ค. ํฌํฌ๋ ๊ฐ ์ ์ ์ฌ์ด์ ์์ต๋๋ค. ์ ๊ณต๋๋ ์๋ฆฌ๋ฅผ ๋จน๊ธฐ ์ํด์๋ ๋ ๊ฐ์ ํฌํฌ๋ฅผ ๋ชจ๋ ์ฌ์ฉํด์ผํฉ๋๋ค. ์ฒ ํ์๋ ์๊ฐ์ ํ๋ค๊ฐ ๋ฐฐ๊ฐ ๊ณ ํ๋ฉด ์์ ์ ์ข,์ฐ์ ํฌํฌ๋ฅผ ๋ค์ด ์๋ฆฌ๋ฅผ ๋จน์ต๋๋ค. ์ฒ ํ์๋ ์๋ฆฌ๋ฅผ ๋จน์ ํ์๋ ํฌํฌ๋ฅผ ๋ค์ ์๋ฆฌ์ ๋ด๋ ค๋์ต๋๋ค. ์ฒ ํ์๋ ์์ ์ ์ข,์ฐ์ ํฌํฌ๊ฐ ์์๋๋ง ์๋ฆฌ๋ฅผ ๋จน์ ์ ์์ต๋๋ค. ๋ฐ๋ผ์ ๋ ๊ฐ์ ํฌํฌ๋ ์ค์ง ์์ ์ ์ข,์ฐ ์ฒ ํ์๊ฐ ์๊ฐ์ ํ ๋๋ง ์ฌ์ฉํ ์ ์์ต๋๋ค.
You will need a local Cargo installation for this exercise. Copy the code below to a file called src/main.rs
, fill out the blanks, and test that cargo run
does not deadlock:
use std::sync::{mpsc, Arc, Mutex}; use std::thread; use std::time::Duration; struct Fork; struct Philosopher { name: String, // left_fork: ... // right_fork: ... // thoughts: ... } impl Philosopher { fn think(&self) { self.thoughts .send(format!("Eureka! {} has a new idea!", &self.name)) .unwrap(); } fn eat(&self) { // Pick up forks... println!("{} is eating...", &self.name); thread::sleep(Duration::from_millis(10)); } } static PHILOSOPHERS: &[&str] = &["Socrates", "Plato", "Aristotle", "Thales", "Pythagoras"]; fn main() { // Create forks // Create philosophers // Make each of them think and eat 100 times // Output their thoughts }
You can use the following Cargo.toml
:
[package]
name = "dining-philosophers"
version = "0.1.0"
edition = "2021"
๋ฉํฐ์ค๋ ๋ ๋งํฌ ๊ฒ์ฌ๊ธฐ
์๋ก ๋ฐฐ์ด๊ฒ๋ค์ ํ์ฉํด์ ๋ฉํฐ ์ค๋ ๋ ๋งํฌ ๊ฒ์ฌ๊ธฐ๋ฅผ ๋ง๋ญ๋๋ค. ์ด ๊ฒ์ฌ๊ธฐ๋ ์นํ์ด์ง ์์ ์๋ ๋งํฌ๋ค์ด ์ ํจํ์ง ํ์ธํฉ๋๋ค. ๊ทธ๋ฆฌ๊ณ ์ฌ๊ท์ ์ผ๋ก ๋์ผ ๋๋ฉ์ธ์ ๋ค๋ฅธ ๋ชจ๋ ํ์ด์ง๊ฐ ์ ํจํ์ง ํ์ธํฉ๋๋ค.
์ด๋ฅผ ์ํด์ reqwest
์ ๊ฐ์ HTTP ํด๋ผ์ด์ธํธ๊ฐ ํ์ํฉ๋๋ค. ์๋ก์ด ๋ก์ปฌ ํ๋ก์ ํธ๋ฅผ ๋ง๋ค๊ณ reqwest
๋ฅผ ์์กด์ฑ์ ์ถ๊ฐํ์ญ์์:
cargo new link-checker
cd link-checker
cargo add --features blocking,rustls-tls reqwest
๋ง์ผ
cargo add
์ปค๋งจ๋๊ฐerror: no such subcommand
๋ก ์คํจํ๋ค๋ฉดCargo.toml
ํ์ผ์ ์ง์ ์์ ํด๋ ๋ฉ๋๋ค. ์๋์ ์ ์ฒด ์์กด์ฑ ๋ด์ฉ์ด ์์ต๋๋ค.
๋งํฌ๋ฅผ ์ฐพ๊ธฐ ์ํด์ scraper
๋ ์ถ๊ฐํฉ๋๋ค:
cargo add scraper
๋ง์ง๋ง์ผ๋ก ์ค๋ฅ ์ฒ๋ฆฌํ๋ ๋ฐฉ๋ฒ์ผ๋ก thiserror
๋ ์ถ๊ฐํฉ๋๋ค:
cargo add thiserror
๋ชจ๋ cargo add
๊ฐ ๋๋๋ฉด Cargo.toml
์ ์๋ ๋ด์ฉ์ด ์ถ๊ฐ๋ฉ๋๋ค:
[package]
name = "link-checker"
version = "0.1.0"
edition = "2021"
publish = false
[dependencies]
reqwest = { version = "0.11.12", features = ["blocking", "rustls-tls"] }
scraper = "0.13.0"
thiserror = "1.0.37"
์ด์ https://www.google.org/
๊ฐ์ ์น ํ์ด์ง๋ฅผ ํ์ํ ์ ์์ต๋๋ค.
rc/main.rs
ํ์ผ์ ์๋์ ๊ฐ์ต๋๋ค:
use reqwest::{blocking::Client, Url}; use scraper::{Html, Selector}; use thiserror::Error; #[derive(Error, Debug)] enum Error { #[error("request error: {0}")] ReqwestError(#[from] reqwest::Error), #[error("bad http response: {0}")] BadResponse(String), } #[derive(Debug)] struct CrawlCommand { url: Url, extract_links: bool, } fn visit_page(client: &Client, command: &CrawlCommand) -> Result<Vec<Url>, Error> { println!("Checking {:#}", command.url); let response = client.get(command.url.clone()).send()?; if !response.status().is_success() { return Err(Error::BadResponse(response.status().to_string())); } let mut link_urls = Vec::new(); if !command.extract_links { return Ok(link_urls); } let base_url = response.url().to_owned(); let body_text = response.text()?; let document = Html::parse_document(&body_text); let selector = Selector::parse("a").unwrap(); let href_values = document .select(&selector) .filter_map(|element| element.value().attr("href")); for href in href_values { match base_url.join(href) { Ok(link_url) => { link_urls.push(link_url); } Err(err) => { println!("On {base_url:#}: ignored unparsable {href:?}: {err}"); } } } Ok(link_urls) } fn main() { let client = Client::new(); let start_url = Url::parse("https://www.google.org").unwrap(); let crawl_command = CrawlCommand{ url: start_url, extract_links: true }; match visit_page(&client, &crawl_command) { Ok(links) => println!("Links: {links:#?}"), Err(err) => println!("Could not extract links: {err:#}"), } }
์๋ ์ปค๋งจ๋๋ก ์์ค๋ฅผ ์คํํฉ๋๋ค
cargo run
ํ์คํฌ
- ์ค๋ ๋๋ฅผ ์ฌ์ฉํ์ฌ ๋งํฌ๋ฅผ ๋ณ๋ ฌ๋ก ํ์ธํฉ๋๋ค: URL์ ์ฑ๋๋ก ๋ณด๋ด์ ๋ช ๊ฐ์ ์ค๋ ๋๊ฐ URL์ ๋ณ๋ ฌ๋ก ์ฒดํฌํ๋๋ก ํฉ๋๋ค.
www.google.org
๋๋ฉ์ธ์ ๋ชจ๋ ํ์ด์ง๋ฅผ ์ฌ๊ท์ ์ผ๋ก ํ์ธํ๊ธฐ ์ํด ์ฝ๋๋ฅผ ํ์ฅํด์ ์์ฑํฉ๋๋ค: ์ฐจ๋จ๋นํ์ง ์๋๋ก 100ํ์ด์ง ์ ๋๋ก ์ ํ์ ๋์๊ธฐ ๋ฐ๋๋๋ค.
Async Rust
โAsyncโ๋ ๋ธ๋ญ๋ (๋ ์ด์ ์งํํ ์ ์์) ๋๊น์ง ๊ฐ ์์ ์ ์คํํ ๋ค์ ์งํํ ์ค๋น๊ฐ ๋ ๋ค๋ฅธ ์์ ์ผ๋ก ์ ํํ์ฌ ์ฌ๋ฌ ์์ ์ ๋์์ ์คํํ๋ ๋์ ์คํ ๋ชจ๋ธ์ ๋๋ค. ์ด ๋ชจ๋ธ์ ์ฌ์ฉํ๋ฉด ์ ํ๋ ์์ ์ค๋ ๋์์ ๋ ๋ง์ ์์ ์ ์คํํ ์ ์์ต๋๋ค. ์ด๋, ํ ์์ ์ ์ ์งํ๊ณ ์ํํ๋๋ฐ ํ์ํ ์ค๋ฒํค๋๊ฐ (์ค๋ ๋์ ๋นํด) ๋งค์ฐ ๋ฎ๊ณ ์ด์์ฒด์ ๊ฐ ์ฌ๋ฌ I/O๋ค์์ ํ์ฌ ์งํ ๊ฐ๋ฅํ I/O๋ค์ ํจ๊ณผ์ ์ผ๋ก ์๋ณํด ์ฃผ๋ ํ๋ฆฌ๋ฏธํฐ๋ธ๋ฅผ ์ ๊ณตํ๊ธฐ ๋๋ฌธ์ ๋๋ค.
Rust์ ๋น๋๊ธฐ ์์ ์ โfuturesโ๋ฅผ ๊ธฐ๋ฐ์ผ๋ก ํ๋ฉฐ ์ด๋ ๋ฏธ๋์ ์๋ฃ๋ ์ ์๋ ์์ ์ ๋ํ๋ ๋๋ค. Futures๋ ์๋ฃ๋์๋ค๋ ์ ํธ๋ฅผ ๋ณด๋ผ ๋๊น์ง โํด๋งโ๋ฉ๋๋ค.
Futures๋ ๋น๋๊ธฐ ๋ฐํ์์ ์ํด ํด๋ง๋๋ฉฐ, ๋น๋๊ธฐ ๋ฐํ์์๋ ์ฌ๋ฌ ๋ค์ํ ์ข ๋ฅ๊ฐ ์์ต๋๋ค.
Comparisons
-
ํ์ด์ฌ์๋
asyncio
๋ผ๋ ์ ์ฌํ ๋ชจ๋ธ์ด ์์ต๋๋ค. ๊ทธ๋ฌ๋ ํ์ด์ฌ์Future
ํ์ ์ ์ฝ๋ฐฑ ๊ธฐ๋ฐ์ด๋ฉฐ ํด๋ง๋์ง ์์ต๋๋ค. ํ์ด์ฌ์ผ๋ก ๋น๋๊ธฐ ํ๋ก๊ทธ๋๋ฐ์ ํ ๋์๋, Rust์์ ๋ฐํ์์ด ๋ด๋ถ์ ์ผ๋ก ํด ์ฃผ๋ ๊ฒ๊ณผ ์ ์ฌํ, โ๋ฃจํโ๋ฅผ ๋ช ์์ ์ผ๋ก ์ฌ์ฉํด์ผ ํฉ๋๋ค. -
์๋ฐ์คํฌ๋ฆฝํธ์
Promise
๋ ๋น์ทํ์ง๋ง ์ญ์ ์ฝ๋ฐฑ ๊ธฐ๋ฐ์ ๋๋ค. ์๋ฐ์คํฌ๋ฆฝํธ์์๋ ์ด๋ฒคํธ ๋ฃจํ๊ฐ๋ฐํ์ ์์ง์์ ๊ตฌํ๋๋ฏ๋กPromise
๊ฐ ์ฒ๋ฆฌ๋๋ ์ธ๋ถ ๊ณผ์ ์ด ์จ๊ฒจ์ง๋๋ค.
async
/await
๊ฒ์์ ๋ณด์์ ๋, ๋น๋๊ธฐ Rust ์ฝ๋๋ ์ผ๋ฐ์ ์ธ ์ ์ฐจ์ ์ฝ๋์ ๋งค์ฐ ์ ์ฌํฉ๋๋ค.
use futures::executor::block_on; async fn count_to(count: i32) { for i in 1..=count { println!("Count is: {i}!"); } } async fn async_main(count: i32) { count_to(count).await; } fn main() { block_on(async_main(10)); }
ํค ํฌ์ธํธ:
-
Rust ๋น๋๊ธฐ ๋ฌธ๋ฒ์ ๋ณด์ฌ์ฃผ๋ ๊ฐ๋จํ ์์์ ๋๋ค. ์ฌ๊ธฐ์๋ ์ค๋ ์คํ๋๋ ์์ ์ด๋, ์ค์ ๋ก ๋์์ ์ํ๋๋ ๊ฒ๋ค์ ์์ต๋๋ค.
-
async
ํจ์์ ๋ฆฌํด ํ์ ์ ๋ฌด์์ธ๊ฐ์?main
์์ `let future: () = async_main(10);์ ์ฌ์ฉํ์ฌ ํ์ ์ ํ์ธํ์ธ์.
-
โasyncโ ํค์๋๋ ๋ฌธ๋ฒ ์คํ(syntactic sugar)์ ๋๋ค. ์ปดํ์ผ๋ฌ๊ฐ ๋ฆฌํด ํ์ ์ future๋ก ๋ฐ๊ฟ๋๋ค.
-
main
์ ๋น๋๊ธฐ ํจ์๋ก ๋ง๋ค์๋ ์์ต๋๋ค. ๋ง์ฝ ๊ทธ๋ ๊ฒ ํ ๊ฒฝ์ฐ ์ปดํ์ผ๋ฌ๋ ๋ฆฌํด ํ์ ์ธ future๋ฅผ ์ด๋ป๊ฒ ์ฌ์ฉํ ์ง ๋ชจ๋ฅด๊ธฐ ๋๋ฌธ์ ๋๋ค. -
๋น๋๊ธฐ ์ฝ๋๋ฅผ ์คํํ๋ ค๋ฉด ์คํ์(executor)๊ฐ ํ์ํฉ๋๋ค.
block_on
์คํ์๋ ์ ๊ณต๋ future๊ฐ ์๋ฃ๋ ๋๊น์ง ํ์ฌ ์ค๋ ๋๋ฅผ ๋ธ๋กํฉ๋๋ค. -
.await
๋ ๋ค๋ฅธ ์์ ์ด ์๋ฃ๋ ๋๊น์ง ๋น๋๊ธฐ์ ์ผ๋ก ๋๊ธฐํฉ๋๋ค.block_on
๊ณผ ๋ฌ๋ฆฌ.await
๋ ํ์ฌ ์ค๋ ๋๋ฅผ ๋ธ๋กํ์ง ์์ต๋๋ค. -
.await
๋async
ํจ์(๋๋ ๋์ค์ ์๊ฐ๋async
๋ธ๋ก) ์์์๋ง ์ฌ์ฉํ ์ ์์ต๋๋ค.
Future
Future
๋ ํธ๋ ์์
๋๋ค.์ด ํธ๋ ์์ ์์ง ์๋ฃ๋์ง ์์์ ์๋ ์๋ ์์
์ ๋ํ๋
๋๋ค. Future๋ poll
ํจ์๋ฅผ ํตํด ํด๋ง๋ ์ ์์ผ๋ฉฐ, ์ด ํจ์๋ Poll
์ ๋ฐํํฉ๋๋ค.
#![allow(unused)] fn main() { use std::pin::Pin; use std::task::Context; pub trait Future { type Output; fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output>; } pub enum Poll<T> { Ready(T), Pending, } }
๋น๋๊ธฐ ํจ์๋ impl Future
๋ฅผ ๋ฐํํฉ๋๋ค. ์๋ก์ด ํ์
์ ๋ง๋ค๊ณ ์ด ํ์
์ด Future
๋ฅผ ๊ตฌํํ๊ฒ ํ ์๋ ์์ง๋ง ์ผ๋ฐ์ ์ด์ง๋ ์์ต๋๋ค. ์๋ฅผ ๋ค์ด tokio::spawn
๊ฐ ๋ฆฌํดํ๋ JoinHandle
์ Future
๋ฅผ ๊ตฌํํ๋ฉฐ, ์ด๋ฅผ ํตํด ์์ฑ๋ ์ค๋ ๋์ joinํ ์ ์์ต๋๋ค.
Future์ .await
๋ฅผ ํธ์ถํ๋ฉด, ํด๋น Future๊ฐ ์ค๋น๋ ๋๊น์ง ํ์ฌ ๋น๋๊ธฐ ํจ์๊ฐ ์ผ์ ์ค์ง๋ฉ๋๋ค. ๊ทธ๋ฐ ๋ค์ Future๊ฐ ์ค๋น๊ฐ ๋๋ฉด, ๊ทธ ๊ฐ์ด .await
๊ตฌ๋ฌธ์ ๊ฐ์ด ๋ฉ๋๋ค.
-
Future
์Poll
ํ์ ์ ์ค์ ์ ์๋ ์์ ๋ณด์ด๋ ๊ทธ๋๋ก ์ ๋๋ค. ๋งํฌ๋ฅผ ํด๋ฆญํ๋ฉด Rust ๋ฌธ์์์ ํ ๋ฒ ๋ ํ์ธํ ์ ์์ต๋๋ค. -
๋ณธ ๊ฐ์์ ๋ชฉ์ ์ ๋น๋๊ธฐ ์ฝ๋๋ฅผ ์์ฑํ๋๋ฐ ์๊ธฐ ๋๋ฌธ์, ์๋ก์ด ๋น๋๊ธฐ ํ๋ฆฌ๋ฏธํฐ๋ธ๋ฅผ ๋ง๋๋๋ฐ ํ์ํ
Pin
๊ณผContext
๋ ๋ค๋ฃจ์ง ์์ต๋๋ค. ์ด๋ค์ ๋ํด ๊ฐ๋จํ ์ค๋ช ํ์๋ฉด:-
Context
๋ฅผ ์ฌ์ฉํ๋ฉด Future๊ฐ ์ด๋ฒคํธ๊ฐ ๋ฐ์ํ ๋ ๋ค์ ํด๋ง๋๋๋ก ์์ฝํ ์ ์์ต๋๋ค. -
โPinโ์ ์ฌ์ฉํ๋ฉด ๋ฉ๋ชจ๋ฆฌ์์ Future์ ์์น๊ฐ ๊ณ ์ ๋๊ธฐ ๋๋ฌธ์ ํด๋น future์ ํฌ์ธํฐ๊ฐ ํญ์ ์ ํจํ๊ฒ ์ ์ง๋ฉ๋๋ค. ์ด๋
.await
ํ์ ์ฐธ์กฐ๋ฅผ ์ ํจํ ์ํ๋ก ์ ์งํ๊ธฐ ์ํด ํ์ํฉ๋๋ค.
-
๋น๋๊ธฐ ๋ฐํ์๋ค
๋น๋๊ธฐ _๋ฐํ์_์ ๋ฆฌ์กํฐ (๋น๋๊ธฐ์ ์์ ์คํ์ ์ง์)์ ์คํ์ (futures๋ฅผ ์คํ)์ ๋ ๊ฐ์ง ์ญํ ์ ํฉ๋๋ค. Rust ์ธ์ด ์์ฒด์์ ๊ธฐ๋ณธ ์ ๊ณตํ๋ ๋น๋๊ธฐ ๋ฐํ์์ ์์ต๋๋ค. ๊ทธ๋ฌ๋ ๋ค์๊ณผ ๊ฐ์ ๋น๋๊ธฐ ๋ฐํ์ ํฌ๋ ์๋ค์ด ์์ต๋๋ค.
- Tokio: performant, with a well-developed ecosystem of functionality like Hyper for HTTP or Tonic for gRPC.
- async-std: aims to be a โstd for asyncโ, and includes a basic runtime in
async::task
. - smol: simple and lightweight
์ฌ๋ฌ ๋๊ท๋ชจ ์ ํ๋ฆฌ์ผ์ด์ ์๋ ์์ฒด ๋ฐํ์์ด ์๋ ๊ฒฝ์ฐ๋ ์์ต๋๋ค. ์๋ค ๋ค์ด Fuchsia๊ฐ ์์ต๋๋ค.
-
Rust ํ๋ ์ด๊ทธ๋ผ์ด๋์์๋ ์์ ๋์ด๋ ๋น๋๊ธฐ ๋ฐํ์ ์ค์์ Tokio๋ง ์ฌ์ฉํ ์ ์์ต๋๋ค. ๋ํ Rust ํ๋ ์ด๊ทธ๋ผ์ด๋๋ I/O๋ฅผ ํ์ฉํ์ง ์์ผ๋ฏ๋ก async๋ฅผ ๊ฐ์ง๊ณ ํ ์ ์๋ ๋ง์ ํฅ๋ฏธ๋ก์ด ์์ ๋ค์ด ๋ถ๊ฐ๋ฅ ํฉ๋๋ค.
-
Futures๋ ์คํ์๊ฐ ํด๋งํ์ง ์๋ ํ ์๋ฌด๊ฒ๋ ํ์ง ์๋๋ค๋ ์ ์์(I/O ์์ ์กฐ์ฐจ ์์ํ์ง ์์) โ๋นํ์ฑโ ์ํ์ ๋๋ค. ์ด๋ ์ฌ์ฉ๋์ง ์๋ ๊ฒฝ์ฐ์๋ ์๋ฃ๋ ๋ ๊น์ง ์คํ๋๋, ์๋ฐ ์คํฌ๋ฆฝํธ์ promise์ ๋ค๋ฆ ๋๋ค.
Tokio
Tokio๋ ๋ค์์ ์ ๊ณตํฉ๋๋ค.
- ๋น๋๊ธฐ ์ฝ๋ ์คํ์ ์ํ ๋ฉํฐ์ค๋ ๋ ๋ฐํ์
- ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ๋น๋๊ธฐ ๋ฒ์
- ๋๊ท๋ชจ ๋ผ์ด๋ธ๋ฌ๋ฆฌ ์ํ๊ณ
use tokio::time; async fn count_to(count: i32) { for i in 1..=count { println!("Count in task: {i}!"); time::sleep(time::Duration::from_millis(5)).await; } } #[tokio::main] async fn main() { tokio::spawn(count_to(10)); for i in 1..5 { println!("Main task: {i}"); time::sleep(time::Duration::from_millis(5)).await; } }
-
์ด์
tokio::main
๋งคํฌ๋ก๋ฅผ ์ฌ์ฉํ๋ฉดmain
์ ๋น๋๊ธฐ๋ก ๋ง๋ค ์ ์์ต๋๋ค. -
spawn
ํจ์๋ ๋์ ์คํ๋๋ ์๋ก์ด โ์์ โ์ ๋ง๋ญ๋๋ค. -
์ฐธ๊ณ :
spawn
์Future
๋ฅผ ์ธ์๋ก ๋ฐ์ต๋๋ค. ๋๋ฌธ์count_to
์.await
๋ฅผ ํธ์ถํ์ง ์๋ ์ ์ ์ฃผ๋ชฉํ์ธ์.
์ฌํ ํ์ต:
-
count_to
๊ฐ 10์ ๋๋ฌํ์ง ์๋ ๊ฒฝ์ฐ๊ฐ ๋ง์๋ฐ ๊ทธ ์ด์ ๋ ๋ฌด์์ผ๊น์? ์ด๋ ๋น๋๊ธฐ์ ์ธ ์ทจ์๋ฅผ ๋ณด์ฌ์ฃผ๋ ์์ ๋๋ค.tokio::spawn
์ด ๋ฆฌํดํ๋ ๊ฒ์ ์๋ฃ๋ ๋๊น์ง ๊ธฐ๋ค๋ฆฌ๋๋ก ๋๊ธฐํ๋๋ฐ ์ฌ์ฉ๋๋ ํธ๋ค์ ๋๋ค. -
tokio::spawn
๋์count_to(10).await
๋ฅผ ์ฌ์ฉํด ๋ณด์ธ์. -
tokio::spawn
์์ ๋ฐํ๋ ์์ ์await
ํด ๋ณด์ธ์.
ํ์คํฌ
Rust์ ํ์คํฌ(์์ ) ์์คํ ์ ๊ฒฝ๋ ์ค๋ ๋ฉ์ ํ ์ข ๋ฅ๋ก ๋ณผ ์ ์์ต๋๋ค.
ํ๋์ ์์
์๋, ์คํ์๊ฐ ์ด ์์
์ ์งํํ๊ธฐ ์ํด ๊ณ์ ํด๋งํ๋, ์ต์์ future๊ฐ ํ ๊ฐ ์์ต๋๋ค. ์ด future์๋ poll
๋ฉ์๋๊ฐ ํด๋งํ๋ ์ค์ฒฉ๋ future๊ฐ ํ ๊ฐ ์ด์ ์์ ์ ์์ต๋๋ค. ์ด๋ฌํ ์ค์ฒฉ๋ future๋ ์ผ๋ฐ์ ์ธ ํจ์ ํธ์ถ ์คํํ๊ณ ๋น์ทํ ์ญํ ์ ํฉ๋๋ค. ํ ์์
์์์ ์ฌ๋ฌ ์์ future๋ค์ ํด๋งํ๋ฉด, ํ์ด๋จธ๋ฅผ ์ผ๋ ๊ฒ๊ณผ ์ด๋ค I/O์์
์ ๋์์ ์ํ์ํจ ํ ํ์ด๋จธ์ I/O ์ค ๋จผ์ ๋๋๋ ๊ฒ์ ๊ธฐ๋ค๋ฆฌ๋ ๊ฒ๊ณผ ๊ฐ์๋์์ฑ๋ ๊ตฌํํ ์ ์์ต๋๋ค.
use tokio::io::{self, AsyncReadExt, AsyncWriteExt}; use tokio::net::TcpListener; #[tokio::main] async fn main() -> io::Result<()> { let listener = TcpListener::bind("127.0.0.1:6142").await?; println!("listening on port 6142"); loop { let (mut socket, addr) = listener.accept().await?; println!("connection from {addr:?}"); tokio::spawn(async move { if let Err(e) = socket.write_all(b"Who are you?\n").await { println!("socket error: {e:?}"); return; } let mut buf = vec![0; 1024]; let reply = match socket.read(&mut buf).await { Ok(n) => { let name = std::str::from_utf8(&buf[..n]).unwrap().trim(); format!("Thanks for dialing in, {name}!\n") } Err(e) => { println!("socket error: {e:?}"); return; } }; if let Err(e) = socket.write_all(reply.as_bytes()).await { println!("socket error: {e:?}"); } }); } }
์ด ์์ ๋ฅผ, ๋ก์ปฌ ์ปดํจํฐ์ ๋ง๋ค์ด ๋ src/main.rs
์ ๋ณต์ฌํ๊ณ ๊ฑฐ๊ธฐ์์ ์คํํ์ธ์.
-
์๊ฐ์๋ค์๊ฒ ์ด ์๋ฒ์ ๋ช ๊ฐ์ ํด๋ผ์ด์ธํธ๊ฐ ์ฐ๊ฒฐ๋๋ฉด ์ด ์๋ฒ์ ์ํ๊ฐ ์ด๋ป๊ฒ ๋ณํ ์ง ๊ทธ๋ฆผ์ ๊ทธ๋ ค๋ณด๋๋ก ํ์ธ์. ์ด๋ค ํ์คํฌ๋ค์ด ์๋์ง, ์ด ํ์คํฌ๋ค์ Future๋ ์ด๋ค ์ํ์ ์๋์ง ๋ฌผ์ด๋ด ๋๋ค.
-
async
๋ธ๋ก์ ์ฒ์ ๋ณด๊ฒ ๋์์ต๋๋ค. ์ด๊ฒ์ ํด๋ก์ ์ ๋น์ทํ์ง๋ง ์ธ์๋ฅผ ๋ฐ์ง ์์ต๋๋ค. ๋ฆฌํด ํ์ ์async fn
๊ณผ ๋น์ทํ Future์ ๋๋ค. -
Async ๋ธ๋ก์ ํจ์๋ก ๋ฆฌํฉํฐ๋งํ๊ณ
?
๋ฅผ ์ฌ์ฉํ์ฌ ์ค๋ฅ ์ฒ๋ฆฌ๋ฅผ ๊ฐ์ ํด ๋ด ์๋ค.
๋น๋๊ธฐ ์ฑ๋
์ฌ๋ฌ ํฌ๋ ์ดํธ์์ ๋น๋๊ธฐ ์ฑ๋์ ์ง์ํฉ๋๋ค. ์๋ฅผ ๋ค์ด tokio
์์๋ ์๋์ ๊ฐ์ดํฉ๋๋ค.
use tokio::sync::mpsc::{self, Receiver}; async fn ping_handler(mut input: Receiver<()>) { let mut count: usize = 0; while let Some(_) = input.recv().await { count += 1; println!("Received {count} pings so far."); } println!("ping_handler complete"); } #[tokio::main] async fn main() { let (sender, receiver) = mpsc::channel(32); let ping_handler_task = tokio::spawn(ping_handler(receiver)); for i in 0..10 { sender.send(()).await.expect("Failed to send ping."); println!("Sent {} pings so far.", i + 1); } drop(sender); ping_handler_task.await.expect("Something went wrong in ping handler task."); }
-
์ฑ๋ ํฌ๊ธฐ๋ฅผ
3
์ผ๋ก ๋ณ๊ฒฝํ๊ณ ๋์์ด ์ด๋ป๊ฒ ๋ฐ๋๋์ง ํ์ธํ์ธ์. -
๋น๋๊ธฐ ์ฑ๋์ ์ฌ์ฉํ๊ธฐ ์ํ ์ธํฐํ์ด์ค๋ ์ค์ ๊ณผ์ ์์ ๋ฐฐ์ด
sync
์ฑ๋๊ณผ ๋น์ทํฉ๋๋ค. -
std::mem::drop
ํธ์ถํ๋ ์ค์ ์ญ์ ํด ๋ณด์ธ์. ์ด๋ค ๊ฒฐ๊ณผ๊ฐ ๋ํ๋๋์? ์ด์ ๊ฐ ๋ฌด์์ธ๊ฐ์? -
Flume ํฌ๋ ์ดํธ์๋
sync
์async
,send
์recv
๋ฅผ ๋ชจ๋ ๊ตฌํํ๋ ์ฑ๋์ด ์์ต๋๋ค. ์ด๊ฒ์ IO์ CPU ์ฒ๋ฆฌ ์์ ์ด ๋ง์ ๋ณต์กํ ์ ํ๋ฆฌ์ผ์ด์ ์ ๊ตฌํํ ๋ ๋งค์ฐ ์ ์ฉํฉ๋๋ค. -
async
์ฑ๋์ ์ฌ์ฉํ๋ ๊ฒ์ด ๋ ์ข์ ์ด์ ๋ ์ด๋ฅผ ๋ค๋ฅธfuture
์ ๊ฒฐํฉํ์ฌ ๋ณต์กํ ์ ์ด ํ๋ฆ์ ๋ง๋ค ์ ์๊ธฐ ๋๋ฌธ์ ๋๋ค.
Futures Control Flow
Future๋ค์ ๊ฒฐํฉํ์ฌ ๊ณ์ฐ ๊ณผ์ ์ ๋์์ฑ์ด ์๋ ํ๋ก์ฐ ๊ทธ๋ํ ํํ๋ก ๋ชจ๋ธ๋ง ํ ์ ์์ต๋๋ค. ์์ ๋ฐฐ์ด, ๊ฐ ํ์คํฌ๊ฐ ๋ ๋ฆฝ์ ์ผ๋ก ์ํ๋๋๋ก ํ๋ ๊ฒ๋ Future๋ค์ ๊ฒฐํฉํ๋ ํ ๋ฐฉ๋ฒ์ผ๋ก ๋ณผ ์ ์์ต๋๋ค.
Join
Join ์ฐ์ฐ์ ๋ชจ๋ future๊ฐ ์ค๋น๋ ๋๊น์ง ๊ธฐ๋ค๋ฆฐ ํ, ๊ฐ future์ ๊ฒฐ๊ณผ๊ฐ์ ๋ด์ ์ปฌ๋ ์
์ ๋ฆฌํดํฉ๋๋ค. ์ด๋ ์๋ฐ์คํฌ๋ฆฝํธ์ Promise.all
์ด๋ ํ์ด์ฌ์ asyncio.gather
์ ์ ์ฌํฉ๋๋ค.
use anyhow::Result; use futures::future; use reqwest; use std::collections::HashMap; async fn size_of_page(url: &str) -> Result<usize> { let resp = reqwest::get(url).await?; Ok(resp.text().await?.len()) } #[tokio::main] async fn main() { let urls: [&str; 4] = [ "https://google.com", "https://httpbin.org/ip", "https://play.rust-lang.org/", "BAD_URL", ]; let futures_iter = urls.into_iter().map(size_of_page); let results = future::join_all(futures_iter).await; let page_sizes_dict: HashMap<&str, Result<usize>> = urls.into_iter().zip(results.into_iter()).collect(); println!("{:?}", page_sizes_dict); }
์ด ์์ ๋ฅผ, ๋ก์ปฌ ์ปดํจํฐ์ ๋ง๋ค์ด ๋ src/main.rs
์ ๋ณต์ฌํ๊ณ ๊ฑฐ๊ธฐ์์ ์คํํ์ธ์.
-
์๋ก ๋ค๋ฅธ ํ์ ์ ๊ฐ์ง๋ ์ฌ๋ฌ ์ฌ๋ฌ futures๋ค์ joinํ๊ณ ์ ํ ๊ฒฝ์ฐ
std::future::join!
์ ์ฌ์ฉํ ์ ์์ต๋๋ค. ์ด ๋งคํฌ๋ก๋ฅผ ์ฌ์ฉํ๋ ค๋ฉด futures๊ฐ ๋ช ๊ฐ๋ ์์์ง ์ปดํ์ผ ํ ๋ ์์์ผ ํ๋ค๋ ์ ์ ์ฃผ์ํ์ธ์. ์ด ๋งคํฌ๋ก๋ ์ง๊ธ์ โfuturesโ ํฌ๋ ์ดํธ์ ์์ผ๋ฉฐ ๊ณง ์์ ํ ๋์ดstd::future
์ ํฌํจ๋ ์์ ์ ๋๋ค. -
โjoinโ์ ์ํ์ฑ์ futures๋ค ์ค ํ๋๊ฐ ์์ ๋๋์ง ์์ ์๋ ์๋ค๋ ๊ฒ์ ๋๋ค. ๊ทธ๋ฌ๋ฉด ํ๋ก๊ทธ๋จ์ด ๋์ด์ ์งํ์ ๋ชปํ๊ณ ๋ฉ์ถฐ์์(stall) ์ ์์ต๋๋ค.
-
join_all
์join!
๊ณผ ๊ฒฐํฉํ์ฌ http ์๋น์ค์ ๋ฐ์ดํฐ๋ฒ ์ด์ค์ ๋ํ ๋ชจ๋ ์์ฒญ๋ค์ ํ๊บผ๋ฒ์ ์งํ์ํฌ ์๋ ์์ต๋๋ค.futures::join!
์ ์ฌ์ฉํ์ฌtokio::time::sleep
์ future์ ์ถ๊ฐํด ๋ณด์ธ์. ์ด๊ฑด ํ์์์์ ๊ตฌํํ๋ ๊ฒ์ด ์๋์ ์ฃผ์ํ์ธ์. ์ค์ ๋ก, ํ์์์์ ๋ค์ ์ฅ์์ ์ค๋ช ํ๋select!
๋ฅผ ์ฌ์ฉํด์ ๊ตฌํํด์ผ ํฉ๋๋ค. ์ฌ๊ธฐ์๋tokio::time::sleep
์ ์ฌ์ฉํ ๊ฒ์ ๋จ์ํjoin!
์ ๋์์ ์ค๋ช ํ๊ธฐ ์ํจ์ ๋๋ค.
Select
Select ์ฐ์ฐ์ ์ฌ๋ฌ future๋ค ๋ชจ๋์ ๋ํด์ ์ค๋น๋ ๋ ๊น์ง ๊ธฐ๋ค๋ฆฌ๋ค๊ฐ, ๊ทธ ์ค ์ด๋ค ํ future๊ฐ ์ต์ด๋ก ์ค๋น ์ํ๊ฐ ๋๋ฉด ํด๋น future์ ๊ฒฐ๊ณผ๊ฐ์ ๋ฆฌํดํฉ๋๋ค. ์ด๊ฒ์ ์๋ฐ์คํฌ๋ฆฝํธ์์์ Promise.race
์ ๋น์ทํฉ๋๋ค. ํ์ด์ฌ์์๋ผ๋ฉด asyncio.wait(task_set, return_when=asyncio.FIRST_COMPLETED)
๊ฐ ํ๋ ๋์๊ณผ ๋น์ทํฉ๋๋ค.
select!
์์๋, match
๋ฌธ๊ณผ ๋น์ทํ๊ฒ, pattern = future => statement
ํํ์ ๋ธ๋์น(arm) ๋ค์ด ์์ต๋๋ค. ์ด๋ค โfutureโ๊ฐ ์งํ ๊ฐ๋ฅ ์ํ๊ฐ ๋๋ฉด โ๊ทธ future
์ ๊ฒฐ๊ณผ๊ฐ์ด pattern
์ผ๋ก ๋ฐ์ธ๋ฉ ๋๋ฉฐ, ๊ทธ ์ํ์์ `statementโ๊ฐ ์ํ๋ฉ๋๋ค.
use tokio::sync::mpsc::{self, Receiver}; use tokio::time::{sleep, Duration}; #[derive(Debug, PartialEq)] enum Animal { Cat { name: String }, Dog { name: String }, } async fn first_animal_to_finish_race( mut cat_rcv: Receiver<String>, mut dog_rcv: Receiver<String>, ) -> Option<Animal> { tokio::select! { cat_name = cat_rcv.recv() => Some(Animal::Cat { name: cat_name? }), dog_name = dog_rcv.recv() => Some(Animal::Dog { name: dog_name? }) } } #[tokio::main] async fn main() { let (cat_sender, cat_receiver) = mpsc::channel(32); let (dog_sender, dog_receiver) = mpsc::channel(32); tokio::spawn(async move { sleep(Duration::from_millis(500)).await; cat_sender .send(String::from("Felix")) .await .expect("Failed to send cat."); }); tokio::spawn(async move { sleep(Duration::from_millis(50)).await; dog_sender .send(String::from("Rex")) .await .expect("Failed to send dog."); }); let winner = first_animal_to_finish_race(cat_receiver, dog_receiver) .await .expect("Failed to receive winner"); println!("Winner is {winner:?}"); }
-
In this example, we have a race between a cat and a dog.
first_animal_to_finish_race
listens to both channels and will pick whichever arrives first. Since the dog takes 50ms, it wins against the cat that take 500ms. -
You can use
oneshot
channels in this example as the channels are supposed to receive only onesend
. -
Try adding a deadline to the race, demonstrating selecting different sorts of futures.
-
Note that
select!
drops unmatched branches, which cancels their futures. It is easiest to use when every execution ofselect!
creates new futures.- An alternative is to pass
&mut future
instead of the future itself, but this can lead to issues, further discussed in the pinning slide.
- An alternative is to pass
async/await์์ ์ฃผ์ํด์ผํ ํจ์
Async์ await๋ ๋์ ๋น๋๊ธฐ ํ๋ก๊ทธ๋๋ฐ์ ์ํ ํธ๋ฆฌํ๊ณ ํจ์จ์ ์ธ ์ถ์ํ๋ฅผ ์ ๊ณตํฉ๋๋ค. ํ์ง๋ง Rust์ async/await ๋ชจ๋ธ์๋ ๋ฌธ์ ๋ ์์ต๋๋ค. ์ด ์ฅ์์ ๋ช ๊ฐ์ง ์๋ฅผ ์ดํด๋ณด๊ฒ ์ต๋๋ค.
์คํ์(executor)๋ฅผ ๋ธ๋ก์ํด
๋๋ถ๋ถ์ ๋น๋๊ธฐ ๋ฐํ์์ IO ์์ ๋ง ๋์์ ์คํ๋๋๋ก ํ์ฉํฉ๋๋ค. ์ฆ, CPU๋ฅผ ๋ธ๋ญํ๋ ํ์คํฌ๊ฐ ์๋ ๊ฒฝ์ฐ, ์ด๋ ์คํ์(executor)๋ฅผ ๋ธ๋ญํ๊ฒ ๋๋ฉฐ, ๊ทธ ๊ฒฐ๊ณผ๋ก ๋ค๋ฅธ ํ์คํฌ๊ฐ ์คํ๋์ง ์์ต๋๋ค. ์ด ๋ฌธ์ ๋ฅผ ํด๊ฒฐํ๋ ๊ฐ๋จํ ๋ฐฉ๋ฒ์, ํญ์ async๋ฅผ ์ง์ํ๋ ๋ฉ์๋๋ฅผ ์ฌ์ฉํ๋ ๊ฒ์ ๋๋ค.
use futures::future::join_all; use std::time::Instant; async fn sleep_ms(start: &Instant, id: u64, duration_ms: u64) { std::thread::sleep(std::time::Duration::from_millis(duration_ms)); println!( "future {id} slept for {duration_ms}ms, finished after {}ms", start.elapsed().as_millis() ); } #[tokio::main(flavor = "current_thread")] async fn main() { let start = Instant::now(); let sleep_futures = (1..=10).map(|t| sleep_ms(&start, t, t * 10)); join_all(sleep_futures).await; }
-
์ฝ๋๋ฅผ ์คํํ์ฌ sleep๋ค์ด ๋์์ ์งํ๋์ง ์๊ณ ์์ฐจ์ ์ผ๋ก์ผ๋ก ์งํ๋๋์ง ํ์ธํ์ธ์.
-
flavor
๋ฅผ"current_thread"
๋ก ์ค์ ํ๋ฉด ๋ชจ๋ ํ์คํฌ๊ฐ ํ๋์ ์ค๋ ๋์์ ์ํ๋ฉ๋๋ค. ์ด๋ ๊ฒ ํ๋ฉด ๋ฌธ์ ์ํฉ์ด ๋ ๋ถ๋ช ํ ๋๋ฌ๋ฉ๋๋ค. ๊ทธ๋ฌ๋ ์ด ๋ฒ๊ทธ๋ ๋ฉํฐ์ค๋ ๋์ธ ๊ฒฝ์ฐ์๋ ์ฌ์ ํ ์กด์ฌํฉ๋๋ค. -
std::thread::sleep
์tokio::time::sleep
์ผ๋ก ๋ฐ๊พธ๊ณ ๊ทธ ๊ฒฐ๊ณผ๋ฅผawait
ํด ๋ณด์ธ์. -
๋ ๋ค๋ฅธ ํด๊ฒฐ ๋ฐฉ๋ฒ์
tokio::task::spawn_blocking
์ ๋๋ค. ์ด๋ ์ค์ ์ค๋ ๋๋ฅผ ์์ฑํ๊ณ , ๊ทธ ์ค๋ ๋์ ๋ํ ํธ๋ค์ future๋ก ๋ณํํจ์ผ๋ก์จ ์คํ์๊ฐ ๋ธ๋ก๋๋ ๊ฒ์ ๋ง์ต๋๋ค. -
ํ์คํฌ๋ฅผ OS ์ค๋ ๋๋ผ๊ณ ์๊ฐํ๋ฉด ์ ๋ฉ๋๋ค. ํ์คํฌ์ OS์ค๋ ๋๋ ์ผ๋์ผ ๋งคํ ๊ด๊ณ์ ์์ง ์์ต๋๋ค. ๋๋ถ๋ถ์ ์คํ์๋ ํ๋์ OS ์ค๋ ๋์์ ์ต๋ํ ๋ง์ ํ์คํฌ๋ฅผ ์ํํ๋๋ก ์ค๊ณ๋์ด ์์ต๋๋ค. ์ด์ ์ FFI๋ฅผ ํตํด ๋ค๋ฅธ ๋ผ์ด๋ธ๋ฌ๋ฆฌ์ ์ํธ์์ฉํ ๋ ํนํ ๋ฌธ์ ๊ฐ ๋ฉ๋๋ค. ์๋ฅผ ๋ค์ด, ํด๋น ๋ผ์ด๋ธ๋ฌ๋ฆฌ๊ฐ ์ค๋ ๋ ๋ก์ปฌ ์ ์ฅ์๋ฅผ ์ด์ฉํ๊ฑฐ๋ ํน์ OS ์ค๋ ๋์ ๋งคํ๋ ์ ์์ต๋๋ค(์: CUDA). ์ด๋ฌํ ์ํฉ์์๋
tokio::task::spawn_blocking
์ ์ฌ์ฉํ๋ ๊ฒ์ด ์ข์ต๋๋ค. -
๋๊ธฐํ ๋ฎคํ ์ค๋ฅผ ์ฃผ์ํด์ ์ฌ์ฉํ์ธ์.
.await
์์ ๋ฎคํ ์ค๋ฅผ ์ ์ฉํ๋ฉด ๋ค๋ฅธ ์์ ์ด ์ฐจ๋จ๋ ์ ์์ผ๋ฉฐ ํด๋น ์์ ์ ๋์ผํ ์ค๋ ๋์์ ์คํ ์ค์ผ ์ ์์ต๋๋ค.
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
์ฑ๋์ ์ ์กํ ๋ค๋ฅธ ์์ ์ ์์ฑํ๋ ๊ฒ์ ๋๋ค.
๋น๋๊ธฐ ํธ๋ ์
ํธ๋ ์์ async ๋ฉ์๋๋ฅผ ์ถ๊ฐํ๋ ๊ฒ์ ์์ง ์์ ํ ๋ฒ์ ์ฑ๋์์ ์ง์๋์ง ์์ต๋๋ค(์คํ์ฉ ๊ธฐ๋ฅ์ nightly์ ์กด์ฌํ๋ฉฐ ์กฐ๋ง๊ฐ ์์ ํ ๋ ๊ฒ์ ๋๋ค).
ํฌ๋ ์ดํธ async_trait์ ๋งคํฌ๋ก๋ฅผ ํตํ ํด๊ฒฐ ๋ฐฉ๋ฒ์ ์ ๊ณตํฉ๋๋ค.
use async_trait::async_trait; use std::time::Instant; use tokio::time::{sleep, Duration}; #[async_trait] trait Sleeper { async fn sleep(&self); } struct FixedSleeper { sleep_ms: u64, } #[async_trait] impl Sleeper for FixedSleeper { async fn sleep(&self) { sleep(Duration::from_millis(self.sleep_ms)).await; } } async fn run_all_sleepers_multiple_times(sleepers: Vec<Box<dyn Sleeper>>, n_times: usize) { for _ in 0..n_times { println!("running all sleepers.."); for sleeper in &sleepers { let start = Instant::now(); sleeper.sleep().await; println!("slept for {}ms", start.elapsed().as_millis()); } } } #[tokio::main] async fn main() { let sleepers: Vec<Box<dyn Sleeper>> = vec![ Box::new(FixedSleeper { sleep_ms: 50 }), Box::new(FixedSleeper { sleep_ms: 100 }), ]; run_all_sleepers_multiple_times(sleepers, 5).await; }
-
async_trait
์ ์ฌ์ฉํ๊ธฐ ์ฝ์ง๋ง ์ด๋ฅผ ์ํด ํ์ ๋ฉ๋ชจ๋ฆฌ๋ฅผ ํ ๋นํ๋ค๋ ์ ์ ์ ์ํ์ธ์. ์ด ํ ํ ๋น์๋ ์ฑ๋ฅ ์ค๋ฒํค๋๊ฐ ์์ต๋๋ค. -
async trait
๋ฅผ ์ธ์ด ์ฐจ์์์ ์ง์ํ๋ ๊ฒ๊ณผ ๊ด๋ จ๋ ๋ฌธ์ ๋ ๋งค์ฐ ์ ๋ฌธ์ ์ธ ํ ํฝ์ด๋ฉฐ ๋ฐ๋ผ์ ์ด ๊ฐ์์์ ๋ค๋ฃฐ ๋ด์ฉ์ ์๋๋๋ค. ์ด ๊ฒ์๋ฌผ์ ์ด์ ๊ดํ ๋์ฝ ๋ง์ฌํค์ค์ ์ข์ ์ค๋ช ์ด ์์ผ๋ฏ๋ก ๊ด์ฌ์ด ์๋ค๋ฉด ์ฐธ๊ณ ํ์ธ์. -
์์์ ์๊ฐ ๋์ sleep ํ๋ ์๋ก์ด sleeper ๊ตฌ์กฐ์ฒด๋ฅผ ๋ง๋ค์ด Vec์ ์ถ๊ฐํด ๋ณด์ธ์.
์ทจ์
Dropping a future implies it can never be polled again. This is called cancellation and it can occur at any await
point. Care is needed to ensure the system works correctly even when futures are cancelled. For example, it shouldnโt deadlock or lose data.
use std::io::{self, ErrorKind}; use std::time::Duration; use tokio::io::{AsyncReadExt, AsyncWriteExt, DuplexStream}; struct LinesReader { stream: DuplexStream, } impl LinesReader { fn new(stream: DuplexStream) -> Self { Self { stream } } async fn next(&mut self) -> io::Result<Option<String>> { let mut bytes = Vec::new(); let mut buf = [0]; while self.stream.read(&mut buf[..]).await? != 0 { bytes.push(buf[0]); if buf[0] == b'\n' { break; } } if bytes.is_empty() { return Ok(None) } let s = String::from_utf8(bytes) .map_err(|_| io::Error::new(ErrorKind::InvalidData, "not UTF-8"))?; Ok(Some(s)) } } async fn slow_copy(source: String, mut dest: DuplexStream) -> std::io::Result<()> { for b in source.bytes() { dest.write_u8(b).await?; tokio::time::sleep(Duration::from_millis(10)).await } Ok(()) } #[tokio::main] async fn main() -> std::io::Result<()> { let (client, server) = tokio::io::duplex(5); let handle = tokio::spawn(slow_copy("hi\nthere\n".to_owned(), client)); let mut lines = LinesReader::new(server); let mut interval = tokio::time::interval(Duration::from_millis(60)); loop { tokio::select! { _ = interval.tick() => println!("tick!"), line = lines.next() => if let Some(l) = line? { print!("{}", l) } else { break }, } } handle.await.unwrap()?; Ok(()) }
-
The compiler doesnโt help with cancellation-safety. You need to read API documentation and consider what state your
async fn
holds. -
Unlike
panic
and?
, cancellation is part of normal control flow (vs error-handling). -
The example loses parts of the string.
-
Whenever the
tick()
branch finishes first,next()
and itsbuf
are dropped. -
LinesReader
can be made cancellation-safe by makingbuf
part of the struct:#![allow(unused)] fn main() { struct LinesReader { stream: DuplexStream, bytes: Vec<u8>, buf: [u8; 1], } impl LinesReader { fn new(stream: DuplexStream) -> Self { Self { stream, bytes: Vec::new(), buf: [0] } } async fn next(&mut self) -> io::Result<Option<String>> { // prefix buf and bytes with self. // ... let raw = std::mem::take(&mut self.bytes); let s = String::from_utf8(raw) // ... } } }
-
-
Interval::tick
is cancellation-safe because it keeps track of whether a tick has been โdeliveredโ. -
AsyncReadExt::read
is cancellation-safe because it either returns or doesnโt read data. -
AsyncBufReadExt::read_line
is similar to the example and isnโt cancellation-safe. See its documentation for details and alternatives.
์ฐ์ต๋ฌธ์
To practice your Async Rust skills, we have again two exercises for you:
-
Dining philosophers: we already saw this problem in the morning. This time you are going to implement it with Async Rust.
-
A Broadcast Chat Application: this is a larger project that allows you experiment with more advanced Async Rust features.
After looking at the exercises, you can look at the solutions provided.
Dining Philosophers - Async
See dining philosophers for a description of the problem.
As before, you will need a local Cargo installation for this exercise. Copy the code below to a file called src/main.rs
, fill out the blanks, and test that cargo run
does not deadlock:
use std::sync::Arc; use tokio::time; use tokio::sync::mpsc::{self, Sender}; use tokio::sync::Mutex; struct Fork; struct Philosopher { name: String, // left_fork: ... // right_fork: ... // thoughts: ... } impl Philosopher { async fn think(&self) { self.thoughts .send(format!("Eureka! {} has a new idea!", &self.name)).await .unwrap(); } async fn eat(&self) { // Pick up forks... println!("{} is eating...", &self.name); time::sleep(time::Duration::from_millis(5)).await; } } static PHILOSOPHERS: &[&str] = &["Socrates", "Plato", "Aristotle", "Thales", "Pythagoras"]; #[tokio::main] async fn main() { // Create forks // Create philosophers // Make them think and eat // Output their thoughts }
Since this time you are using Async Rust, youโll need a tokio
dependency. You can use the following Cargo.toml
:
[package]
name = "dining-philosophers-async-dine"
version = "0.1.0"
edition = "2021"
[dependencies]
tokio = {version = "1.26.0", features = ["sync", "time", "macros", "rt-multi-thread"]}
Also note that this time you have to use the Mutex
and the mpsc
module from the tokio
crate.
- Can you make your implementation single-threaded?
์ฑํ ์ ํ๋ฆฌ์ผ์ด์
In this exercise, we want to use our new knowledge to implement a broadcast chat application. We have a chat server that the clients connect to and publish their messages. The client reads user messages from the standard input, and sends them to the server. The chat server broadcasts each message that it receives to all the clients.
For this, we use a broadcast channel on the server, and tokio_websockets
for the communication between the client and the server.
Create a new Cargo project and add the following dependencies:
Cargo.toml
:
[package]
name = "chat-async"
version = "0.1.0"
edition = "2021"
[dependencies]
futures-util = { version = "0.3.28", features = ["sink"] }
http = "0.2.9"
tokio = { version = "1.28.1", features = ["full"] }
tokio-websockets = { version = "0.4.0", features = ["client", "fastrand", "server", "sha1_smol"] }
The required APIs
You are going to need the following functions from tokio
and tokio_websockets
. Spend a few minutes to familiarize yourself with the API.
- StreamExt::next() implemented by
WebsocketStream
: for asynchronously reading messages from a Websocket Stream. - SinkExt::send() implemented by
WebsocketStream
: for asynchronously sending messages on a Websocket Stream. - Lines::next_line(): for asynchronously reading user messages from the standard input.
- Sender::subscribe(): for subscribing to a broadcast channel.
Two binaries
Normally in a Cargo project, you can have only one binary, and one src/main.rs
file. In this project, we need two binaries. One for the client, and one for the server. You could potentially make them two separate Cargo projects, but we are going to put them in a single Cargo project with two binaries. For this to work, the client and the server code should go under src/bin
(see the documentation).
Copy the following server and client code into src/bin/server.rs
and src/bin/client.rs
, respectively. Your task is to complete these files as described below.
src/bin/server.rs
:
use futures_util::sink::SinkExt; use futures_util::stream::StreamExt; use std::error::Error; use std::net::SocketAddr; use tokio::net::{TcpListener, TcpStream}; use tokio::sync::broadcast::{channel, Sender}; use tokio_websockets::{Message, ServerBuilder, WebsocketStream}; async fn handle_connection( addr: SocketAddr, mut ws_stream: WebsocketStream<TcpStream>, bcast_tx: Sender<String>, ) -> Result<(), Box<dyn Error + Send + Sync>> { // TODO: For a hint, see the description of the task below. } #[tokio::main] async fn main() -> Result<(), Box<dyn Error + Send + Sync>> { let (bcast_tx, _) = channel(16); let listener = TcpListener::bind("127.0.0.1:2000").await?; println!("listening on port 2000"); loop { let (socket, addr) = listener.accept().await?; println!("New connection from {addr:?}"); let bcast_tx = bcast_tx.clone(); tokio::spawn(async move { // Wrap the raw TCP stream into a websocket. let ws_stream = ServerBuilder::new().accept(socket).await?; handle_connection(addr, ws_stream, bcast_tx).await }); } }
src/bin/client.rs
:
use futures_util::stream::StreamExt; use futures_util::SinkExt; use http::Uri; use tokio::io::{AsyncBufReadExt, BufReader}; use tokio_websockets::{ClientBuilder, Message}; #[tokio::main] async fn main() -> Result<(), tokio_websockets::Error> { let (mut ws_stream, _) = ClientBuilder::from_uri(Uri::from_static("ws://127.0.0.1:2000")) .connect() .await?; let stdin = tokio::io::stdin(); let mut stdin = BufReader::new(stdin).lines(); // TODO: For a hint, see the description of the task below. }
Running the binaries
Run the server with:
cargo run --bin server
and the client with:
cargo run --bin client
ํ์คํฌ
- Implement the
handle_connection
function insrc/bin/server.rs
.- Hint: Use
tokio::select!
for concurrently performing two tasks in a continuous loop. One task receives messages from the client and broadcasts them. The other sends messages received by the server to the client.
- Hint: Use
- Complete the main function in
src/bin/client.rs
.- Hint: As before, use
tokio::select!
in a continuous loop for concurrently performing two tasks: (1) reading user messages from standard input and sending them to the server, and (2) receiving messages from the server, and displaying them for the user.
- Hint: As before, use
- Optional: Once you are done, change the code to broadcast messages to all clients, but the sender of the message.
๊ฐ์ฌ์ธ์ฌ
Comprehensive Rust ๐ฆ๋ฅผ ์ด์ฉํด ์ฃผ์ ์ ๊ฐ์ฌํฉ๋๋ค. ์ฆ๊ฒ๊ณ ์ ์ตํ ์๊ฐ์ด์๊ธฐ๋ฅผ ๋ฐ๋๋๋ค.
๊ฐ์๊ฐ ์๋ฒฝํ์ง ์์ผ๋ ์ค์๋ ๊ฐ์ ์ ์ด ์๋ค๋ฉด ์ธ์ ๋ ์ง ๊นํ๋ธ๋ก ์ฐ๋ฝ์ฃผ์ธ์.
๋ฌ์คํธ ์ฐธ๊ณ ์๋ฃ
๋ฌ์คํธ ์ปค๋ฎค๋ํฐ๋ ์จ๋ผ์ธ์์ ๊ณ ํ์ง์ ๋ฌด๋ฃ ์์ค๋ฅผ ๋ง๋ค์์ต๋๋ค.
๊ณต์ ๋ฌธ์๋ค
๋ฌ์คํธ ํ๋ก์ ํธ์๋ ์ฐธ์กฐํ ๋งํ ์๋ฃ๊ฐ ๋ง์ต๋๋ค. ์ผ๋ฐ์ ์ธ ๋ด์ฉ์ ๋ค๋ฃจ๋ ๋ช๊ฐ์ง ์ฐธ๊ณ ๋ฌธ์๋ค์ ๋๋ค:
- The Rust Programming Language: ๋ฌ์คํธ์ ๋ํ ๋ฌด๋ฃ ํ์ค ์์ ์ ๋๋ค. ์ธ์ด์ ๋ํ ์์ธํ ์ค๋ช ๊ณผ ์ฌ๋๋ค์ด ๋น๋ ํ ์ ์๋ ๋ช๊ฐ์ง ํ๋ก์ ํธ๋ฅผ ํฌํจํฉ๋๋ค.
- Rust By Example: ์ฌ๋ฌ ์์ ๋ฅผ ํตํด ๋ฌ์คํธ์ ๋ฌธ๋ฒ์ ๋ณด์ฌ์ฃผ๋ฉฐ ๋๋๋ก ์ฝ๋๋ฅผ ํ์ฅํ๋ ์ฝ๊ฐ์ ์ฐ์ต๋ฌธ์ ๋ค์ด ํฌํจ๋์ด ์์ต๋๋ค.
- Rust Standard Library: ๋ฌ์คํธ ํ์ค ๋ผ์ด๋ธ๋ฌ๋ฆฌ ์ ์ฒด ๋ฌธ์์ ๋๋ค.
- The Rust Reference: ๋ฉ๋ชจ๋ฆฌ ๋ชจ๋ธ๋ง๊ณผ ๋ฌ์คํธ ๋ฌธ๋ฒ์ ์ค๋ช ํ๋ ๋ฌธ์์ ๋๋ค.(์์ง ๋ถ์์ ํ๋คํจ)
์ข ๋ ์ ๋ฌธ์ ์ธ ๊ณต์ ๊ฐ์ด๋์ ๋๋ค:
- The Rustonomicon: ์์ ํ์ง ์์ ๋ฌ์คํธ, FFI, rawํฌ์ธํฐ ์์ ์ ๋ค๋ฃน๋๋ค.
- Asynchronous Programming in Rust: ๋ฌ์คํธ ๋ถ์ด ์์ฑ ๋ ์ดํ ๋์ ๋ ์๋ก์ด ๋น๋๊ธฐ ํ๋ก๊ทธ๋๋ฐ ๋ชจ๋ธ์ ๋ค๋ฃน๋๋ค.
- The Embedded Rust Book: ์ด์์ฒด์ ๊ฐ ์๋ ์๋ฒ ๋๋ ์ฅ์น์์์ ๋ฌ์คํธ ์ฌ์ฉ๋ฒ์ ์๊ฐํฉ๋๋ค.
๋น๊ณต์์ ํ์ต ์๋ฃ
๋ฌ์คํธ์ ๋ํ ๊ธฐํ ์๋ด์์ ํํ ๋ฆฌ์ผ์ ์ผ๋ถ์ ๋๋ค:
- Learn Rust the Dangerous Way: C์ธ์ด ํ๋ก๊ทธ๋๋จธ ๊ด์ ์์ ๋ฌ์คํธ๋ฅผ ๋ค๋ฃน๋๋ค.
- Rust for Embedded C Programmers: ์๋ฒ ๋๋ C๊ฐ๋ฐ์(ํ์จ์ด ๊ฐ๋ฐ์)๋ฅผ ์ํ ๋ฌ์คํธ ๊ฐ์ด๋์ ๋๋ค.
- Rust for professionals: ๋ค๋ฅธ ์ธ์ด(C/C++, Java, Python, Javascript)์์ ๋ณ๋ ฌ๋น๊ต๋ฅผ ์ฌ์ฉํ์ฌ ๋ฌ์คํธ ๋ฌธ๋ฒ์ ๋ค๋ฃน๋๋ค.
- Rust on Exercism: ๋ฌ์คํธ๋ฅผ ๋ฐฐ์ฐ๋๋ฐ ๋์์ด ๋๋ 100๊ฐ ์ด์์ ์ฐ์ต๋ฌธ์
- Ferrous Teaching Material: ๋ฌ์คํธ ์ธ์ด์ ๊ธฐ๋ณธ๋ถํฐ ๊ณ ๊ธ์ ์ ๋ถ ๋ค๋ฃจ๋ ์ผ๋ จ์ ์์ ํ๋ ์ ํ ์ด์ , ์น ์ด์ ๋ธ๋ฆฌ, async/await ๊ฐ์ ๋ถ๋ถ๋ ํจ๊ป ๋ค๋ฃน๋๋ค.
- Beginnerโs Series to Rust, Take your first steps with Rust: ์ฒซ๋ฒ์งธ๋ 35๊ฐ์ ์๋ฆฌ์ฆ ์์์ด๋ฉฐ ๋๋ฒ์งธ๋ ๋ฌ์คํธ์ ๋ฌธ๋ฒ๊ณผ ๊ตฌ์กฐ๋ฅผ ๋ค๋ฃจ๋ 11๊ฐ์ ๋ชจ๋ ์ธํธ์ ๋๋ค.
- Learn Rust With Entirely Too Many Linked Lists: ๋ช๊ฐ์ง ์ ํ์ ๋ฆฌ์คํธ ์๋ฃ๊ตฌ์กฐ๋ฅผ ๊ตฌํํด๋ณด๋ฉด์ ๋ฌ์คํธ์ ๋ฉ๋ชจ๋ฆฌ ๊ด๋ฆฌ ๊ท์น๋ค์ ๊น์ด์๊ฒ ํ์ํฉ๋๋ค.
Little Book of Rust Books์์ ๋ ๋ง์ ๋ฌ์คํธ ๋ถ์ ํ์ธํด๋ณด์ธ์.
๋์์ฃผ์ ๋ถ๋ค
์ด ์๋ฃ๋ ๋ง์ ํ๋ฅญํ ๋ฌ์คํธ ๋ฌธ์๋ค์ ๋์์ ๋ฐ์ ์์ฑ๋์์ต๋๋ค. ์ ์ฉํ ์๋ฃ์ ์ ์ฒด ๋ชฉ๋ก์ other resources์์ ์ดํด๋ณด์๊ธฐ ๋ฐ๋๋๋ค.
The material of Comprehensive Rust is licensed under the terms of the Apache 2.0 license, please see LICENSE
for details.
Rust by Example
์ผ๋ถ ์์ ์ ์ฐ์ต๋ฌธ์ ๋ Rust by Example์ ์ฐธ์กฐํ์์ต๋๋ค. ๋ผ์ด์ ์ค ์กฐํญ์ ํฌํจํ์ฌ ์ ์ฅ์์ third_party/rust-by-example/
ํด๋๋ฅผ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค.
Rust on Exercism
์ผ๋ถ ์ฐ์ต๋ฌธ์ ๋ Rust on Exercism์ ์ฐธ์กฐํ์์ต๋๋ค. ๋ผ์ด์ ์ค ์กฐํญ์ ํฌํจํ์ฌ ์ ์ฅ์์ third_party/rust-on-exercism/
ํด๋๋ฅผ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค.
CXX
4์ผ์ฐจ ์คํ ๊ฐ์ ์ค Interoperability with C++์์๋ CXX์ ์ด๋ฏธ์ง๋ฅผ ์ฌ์ฉํ์์ต๋๋ค. ๋ผ์ด์ ์ค ์กฐํญ์ ํฌํจํ์ฌ ์ ์ฅ์์ third_party/cxx/
ํด๋๋ฅผ ์ฐธ์กฐํ์๊ธฐ ๋ฐ๋๋๋ค.
ํด๋ต
์ฐ์ต๋ฌธ์ ์ ํด๋ต์ ๋ค์ ํ์ด์ง์์ ํ์ธํ ์ ์์ต๋๋ค.
๊นํ๋ธ์์ ์ด์ ๋ํด ์์ ๋กญ๊ฒ ์ง๋ฌธํ์๊ณ ๋ ๋์ ์๋ฃจ์ ์ด ์๋ค๋ฉด ์๋ ค์ฃผ์๊ธฐ ๋ฐ๋๋๋ค.
1์ผ์ฐจ ์ค์ ์ฐ์ต๋ฌธ์
๋ฐฐ์ด๊ณผ for
๋ฐ๋ณต๋ฌธ
fn transpose(matrix: [[i32; 3]; 3]) -> [[i32; 3]; 3] { let mut result = [[0; 3]; 3]; for i in 0..3 { for j in 0..3 { result[j][i] = matrix[i][j]; } } return result; } fn pretty_print(matrix: &[[i32; 3]; 3]) { for row in matrix { println!("{row:?}"); } } #[test] fn test_transpose() { let matrix = [ [101, 102, 103], // [201, 202, 203], [301, 302, 303], ]; let transposed = transpose(matrix); assert_eq!( transposed, [ [101, 201, 301], // [102, 202, 302], [103, 203, 303], ] ); } fn main() { let matrix = [ [101, 102, 103], // <-- the comment makes rustfmt add a newline [201, 202, 203], [301, 302, 303], ]; println!("matrix:"); pretty_print(&matrix); let transposed = transpose(matrix); println!("transposed:"); pretty_print(&transposed); }
๋ณด๋์ค ๋ฌธ์
์ฌ์ค ์ด ๋ฌธ์ ๋ ๊ณ ๊ธ ๊ฐ๋
์ด ํ์ํฉ๋๋ค. ์ฌ๋ผ์ด์ค์ ์ฌ๋ผ์ด์ค(slice-of-slices, &[&[i32]]
)๋ฅผ ์
๋ ฅ ํ์
์ผ๋ก ์ฌ์ฉํ๋ฉด ๋ชจ๋ ํฌ๊ธฐ์ ํ๋ ฌ์ ์ฒ๋ฆฌํ ์ ์์๊ฒ ๊ฐ์ต๋๋ค. ํ์ง๋ง ์ค์ ๋ก ํด๋ณด๋ฉด ๊ธ๋ฐฉ ์๋๋ค๋ ๊ฑธ ์ ์ ์์ต๋๋ค. ๋ฐํ๊ฐ์ ์์ ํด์ผ ํ๊ธฐ๋๋ฌธ์ &[&[i32]]
๋ฐํ ํ์
์ผ๋ก ์ฌ์ฉํ ์ ์์ต๋๋ค.
Vec<Vec<i32>>
์ ๊ฐ์ ํ์
์ ์ฌ์ฉํ๋ ค๊ณ ์๋ํ ์๋ ์์ง๋ง ์ญ์ ์ฝ๊ฒ ๋์ง ์์ต๋๋ค. Vec<Vec<i32>>
ํ์
์ &[&[i32]]
๋ก ๋ณํํ๋ ๊ฒ์ด ์ด๋ ต๊ธฐ ๋๋ฌธ์ pretty_print
์ ์ฌ์ฉํ๋๋ฐ ์ด๋ ค์์ด ์์ต๋๋ค.
ํธ๋ ์๋ ์ ๋ค๋ฆญ์ ๋ค๋ฃจ๊ณ ๋๋ฉด std::convert::AsRef
ํธ๋ ์์ ์ฌ์ฉํ์ฌ ์ฌ๋ผ์ด์ค์ฒ๋ผ ์ฌ์ฉ๋ ์ ์๋ ํ์
์ ์ถ์ํํ ์ ์์ต๋๋ค.
use std::convert::AsRef; use std::fmt::Debug; fn pretty_print<T, Line, Matrix>(matrix: Matrix) where T: Debug, // A line references a slice of items Line: AsRef<[T]>, // A matrix references a slice of lines Matrix: AsRef<[Line]> { for row in matrix.as_ref() { println!("{:?}", row.as_ref()); } } fn main() { // &[&[i32]] pretty_print(&[&[1, 2, 3], &[4, 5, 6], &[7, 8, 9]]); // [[&str; 2]; 2] pretty_print([["a", "b"], ["c", "d"]]); // Vec<Vec<i32>> pretty_print(vec![vec![1, 2], vec![3, 4]]); }
๋ํ, ์ฌ๋ผ์ด์ค ํ์ ์ ๊ธธ์ด๋ฅผ ํฌํจํ์ง ์๊ธฐ ๋๋ฌธ์ ํ ๋จ๊ณ ์๋์ ์ฌ๋ผ์ด์ค๋ค์ด ๊ฐ์ ๊ธธ์ด์์ ๋ณด์ฅํ ์ ์์ต๋๋ค. ๋๋ฌธ์ ์ฌ๋ผ์ด์ค ํ์ ์ ๋ณ์์๋ ์๋ชป๋ ํ๋ ฌ์ด ์ ๋ฌ๋ ์ ์์ต๋๋ค.
1์ผ์ฐจ ์คํ ์ฐ์ต๋ฌธ์
๋ฃฌ ์๊ณ ๋ฆฌ์ฆ
pub fn luhn(cc_number: &str) -> bool { let mut digits_seen = 0; let mut sum = 0; for (i, ch) in cc_number.chars().rev().filter(|&ch| ch != ' ').enumerate() { match ch.to_digit(10) { Some(d) => { sum += if i % 2 == 1 { let dd = d * 2; dd / 10 + dd % 10 } else { d }; digits_seen += 1; } None => return false, } } if digits_seen < 2 { return false; } sum % 10 == 0 } fn main() { let cc_number = "1234 5678 1234 5670"; println!( "Is {cc_number} a valid credit card number? {}", if luhn(cc_number) { "yes" } else { "no" } ); } #[test] fn test_non_digit_cc_number() { assert!(!luhn("foo")); assert!(!luhn("foo 0 0")); } #[test] fn test_empty_cc_number() { assert!(!luhn("")); assert!(!luhn(" ")); assert!(!luhn(" ")); assert!(!luhn(" ")); } #[test] fn test_single_digit_cc_number() { assert!(!luhn("0")); } #[test] fn test_two_digit_cc_number() { assert!(luhn(" 0 0 ")); } #[test] fn test_valid_cc_number() { assert!(luhn("4263 9826 4026 9299")); assert!(luhn("4539 3195 0343 6467")); assert!(luhn("7992 7398 713")); } #[test] fn test_invalid_cc_number() { assert!(!luhn("4223 9826 4026 9299")); assert!(!luhn("4539 3195 0343 6476")); assert!(!luhn("8273 1232 7352 0569")); }
Pattern matching
/// An operation to perform on two subexpressions. #[derive(Debug)] enum Operation { Add, Sub, Mul, Div, } /// An expression, in tree form. #[derive(Debug)] enum Expression { /// An operation on two subexpressions. Op { op: Operation, left: Box<Expression>, right: Box<Expression>, }, /// A literal value Value(i64), } /// The result of evaluating an expression. #[derive(Debug, PartialEq, Eq)] enum Res { /// Evaluation was successful, with the given result. Ok(i64), /// Evaluation failed, with the given error message. Err(String), } // Allow `Ok` and `Err` as shorthands for `Res::Ok` and `Res::Err`. use Res::{Err, Ok}; fn eval(e: Expression) -> Res { match e { Expression::Op { op, left, right } => { let left = match eval(*left) { Ok(v) => v, Err(msg) => return Err(msg), }; let right = match eval(*right) { Ok(v) => v, Err(msg) => return Err(msg), }; Ok(match op { Operation::Add => left + right, Operation::Sub => left - right, Operation::Mul => left * right, Operation::Div => { if right == 0 { return Err(String::from("division by zero")); } else { left / right } } }) } Expression::Value(v) => Ok(v), } } #[test] fn test_value() { assert_eq!(eval(Expression::Value(19)), Ok(19)); } #[test] fn test_sum() { assert_eq!( eval(Expression::Op { op: Operation::Add, left: Box::new(Expression::Value(10)), right: Box::new(Expression::Value(20)), }), Ok(30) ); } #[test] fn test_recursion() { let term1 = Expression::Op { op: Operation::Mul, left: Box::new(Expression::Value(10)), right: Box::new(Expression::Value(9)), }; let term2 = Expression::Op { op: Operation::Mul, left: Box::new(Expression::Op { op: Operation::Sub, left: Box::new(Expression::Value(3)), right: Box::new(Expression::Value(4)), }), right: Box::new(Expression::Value(5)), }; assert_eq!( eval(Expression::Op { op: Operation::Add, left: Box::new(term1), right: Box::new(term2), }), Ok(85) ); } #[test] fn test_error() { assert_eq!( eval(Expression::Op { op: Operation::Div, left: Box::new(Expression::Value(99)), right: Box::new(Expression::Value(0)), }), Err(String::from("division by zero")) ); } fn main() { let expr = Expression::Op { op: Operation::Sub, left: Box::new(Expression::Value(20)), right: Box::new(Expression::Value(10)), }; println!("expr: {:?}", expr); println!("result: {:?}", eval(expr)); }
2์ผ์ฐจ ์ค์ ์ฐ์ต๋ฌธ์
๋์๊ด ์ค๊ณ
struct Library { books: Vec<Book>, } struct Book { title: String, year: u16, } impl Book { // This is a constructor, used below. fn new(title: &str, year: u16) -> Book { Book { title: String::from(title), year, } } } // Implement the methods below. Notice how the `self` parameter // changes type to indicate the method's required level of ownership // over the object: // // - `&self` for shared read-only access, // - `&mut self` for unique and mutable access, // - `self` for unique access by value. impl Library { fn new() -> Library { Library { books: Vec::new() } } fn len(&self) -> usize { self.books.len() } fn is_empty(&self) -> bool { self.books.is_empty() } fn add_book(&mut self, book: Book) { self.books.push(book) } fn print_books(&self) { for book in &self.books { println!("{}, published in {}", book.title, book.year); } } fn oldest_book(&self) -> Option<&Book> { // Using a closure and a built-in method: // self.books.iter().min_by_key(|book| book.year) // Longer hand-written solution: let mut oldest: Option<&Book> = None; for book in self.books.iter() { if oldest.is_none() || book.year < oldest.unwrap().year { oldest = Some(book); } } oldest } } fn main() { let mut library = Library::new(); println!( "The library is empty: library.is_empty() -> {}", library.is_empty() ); library.add_book(Book::new("Lord of the Rings", 1954)); library.add_book(Book::new("Alice's Adventures in Wonderland", 1865)); println!( "The library is no longer empty: library.is_empty() -> {}", library.is_empty() ); library.print_books(); match library.oldest_book() { Some(book) => println!("The oldest book is {}", book.title), None => println!("The library is empty!"), } println!("The library has {} books", library.len()); library.print_books(); } #[test] fn test_library_len() { let mut library = Library::new(); assert_eq!(library.len(), 0); assert!(library.is_empty()); library.add_book(Book::new("Lord of the Rings", 1954)); library.add_book(Book::new("Alice's Adventures in Wonderland", 1865)); assert_eq!(library.len(), 2); assert!(!library.is_empty()); } #[test] fn test_library_is_empty() { let mut library = Library::new(); assert!(library.is_empty()); library.add_book(Book::new("Lord of the Rings", 1954)); assert!(!library.is_empty()); } #[test] fn test_library_print_books() { let mut library = Library::new(); library.add_book(Book::new("Lord of the Rings", 1954)); library.add_book(Book::new("Alice's Adventures in Wonderland", 1865)); // We could try and capture stdout, but let us just call the // method to start with. library.print_books(); } #[test] fn test_library_oldest_book() { let mut library = Library::new(); assert!(library.oldest_book().is_none()); library.add_book(Book::new("Lord of the Rings", 1954)); assert_eq!( library.oldest_book().map(|b| b.title.as_str()), Some("Lord of the Rings") ); library.add_book(Book::new("Alice's Adventures in Wonderland", 1865)); assert_eq!( library.oldest_book().map(|b| b.title.as_str()), Some("Alice's Adventures in Wonderland") ); }
2์ผ์ฐจ ์คํ ์ฐ์ต๋ฌธ์
๋ฌธ์์ด๊ณผ ๋ฐ๋ณต์
pub fn prefix_matches(prefix: &str, request_path: &str) -> bool { let mut request_segments = request_path.split('/'); for prefix_segment in prefix.split('/') { let Some(request_segment) = request_segments.next() else { return false; }; if request_segment != prefix_segment && prefix_segment != "*" { return false; } } true // Alternatively, Iterator::zip() lets us iterate simultaneously over prefix // and request segments. The zip() iterator is finished as soon as one of // the source iterators is finished, but we need to iterate over all request // segments. A neat trick that makes zip() work is to use map() and chain() // to produce an iterator that returns Some(str) for each pattern segments, // and then returns None indefinitely. } #[test] fn test_matches_without_wildcard() { assert!(prefix_matches("/v1/publishers", "/v1/publishers")); assert!(prefix_matches("/v1/publishers", "/v1/publishers/abc-123")); assert!(prefix_matches("/v1/publishers", "/v1/publishers/abc/books")); assert!(!prefix_matches("/v1/publishers", "/v1")); assert!(!prefix_matches("/v1/publishers", "/v1/publishersBooks")); assert!(!prefix_matches("/v1/publishers", "/v1/parent/publishers")); } #[test] fn test_matches_with_wildcard() { assert!(prefix_matches( "/v1/publishers/*/books", "/v1/publishers/foo/books" )); assert!(prefix_matches( "/v1/publishers/*/books", "/v1/publishers/bar/books" )); assert!(prefix_matches( "/v1/publishers/*/books", "/v1/publishers/foo/books/book1" )); assert!(!prefix_matches("/v1/publishers/*/books", "/v1/publishers")); assert!(!prefix_matches( "/v1/publishers/*/books", "/v1/publishers/foo/booksByAuthor" )); } fn main() {}
3์ผ์ฐจ ์ค์ ์ฐ์ต๋ฌธ์
๊ฐ๋จํ GUI ๋ผ์ด๋ธ๋ฌ๋ฆฌ
pub trait Widget { /// Natural width of `self`. fn width(&self) -> usize; /// Draw the widget into a buffer. fn draw_into(&self, buffer: &mut dyn std::fmt::Write); /// Draw the widget on standard output. fn draw(&self) { let mut buffer = String::new(); self.draw_into(&mut buffer); println!("{buffer}"); } } pub struct Label { label: String, } impl Label { fn new(label: &str) -> Label { Label { label: label.to_owned(), } } } pub struct Button { label: Label, callback: Box<dyn FnMut()>, } impl Button { fn new(label: &str, callback: Box<dyn FnMut()>) -> Button { Button { label: Label::new(label), callback, } } } pub struct Window { title: String, widgets: Vec<Box<dyn Widget>>, } impl Window { fn new(title: &str) -> Window { Window { title: title.to_owned(), widgets: Vec::new(), } } fn add_widget(&mut self, widget: Box<dyn Widget>) { self.widgets.push(widget); } fn inner_width(&self) -> usize { std::cmp::max( self.title.chars().count(), self.widgets.iter().map(|w| w.width()).max().unwrap_or(0), ) } } impl Widget for Window { fn width(&self) -> usize { // Add 4 paddings for borders self.inner_width() + 4 } fn draw_into(&self, buffer: &mut dyn std::fmt::Write) { let mut inner = String::new(); for widget in &self.widgets { widget.draw_into(&mut inner); } let inner_width = self.inner_width(); // TODO: after learning about error handling, you can change // draw_into to return Result<(), std::fmt::Error>. Then use // the ?-operator here instead of .unwrap(). writeln!(buffer, "+-{:-<inner_width$}-+", "").unwrap(); writeln!(buffer, "| {:^inner_width$} |", &self.title).unwrap(); writeln!(buffer, "+={:=<inner_width$}=+", "").unwrap(); for line in inner.lines() { writeln!(buffer, "| {:inner_width$} |", line).unwrap(); } writeln!(buffer, "+-{:-<inner_width$}-+", "").unwrap(); } } impl Widget for Button { fn width(&self) -> usize { self.label.width() + 8 // add a bit of padding } fn draw_into(&self, buffer: &mut dyn std::fmt::Write) { let width = self.width(); let mut label = String::new(); self.label.draw_into(&mut label); writeln!(buffer, "+{:-<width$}+", "").unwrap(); for line in label.lines() { writeln!(buffer, "|{:^width$}|", &line).unwrap(); } writeln!(buffer, "+{:-<width$}+", "").unwrap(); } } impl Widget for Label { fn width(&self) -> usize { self.label .lines() .map(|line| line.chars().count()) .max() .unwrap_or(0) } fn draw_into(&self, buffer: &mut dyn std::fmt::Write) { writeln!(buffer, "{}", &self.label).unwrap(); } } fn main() { let mut window = Window::new("Rust GUI Demo 1.23"); window.add_widget(Box::new(Label::new("This is a small text GUI demo."))); window.add_widget(Box::new(Button::new( "Click me!", Box::new(|| println!("You clicked the button!")), ))); window.draw(); }
์ ๊ณผ ๋ค๊ฐํ
#[derive(Debug, Copy, Clone, PartialEq, Eq)] pub struct Point { x: i32, y: i32, } impl Point { pub fn new(x: i32, y: i32) -> Point { Point { x, y } } pub fn magnitude(self) -> f64 { f64::from(self.x.pow(2) + self.y.pow(2)).sqrt() } pub fn dist(self, other: Point) -> f64 { (self - other).magnitude() } } impl std::ops::Add for Point { type Output = Self; fn add(self, other: Self) -> Self::Output { Self { x: self.x + other.x, y: self.y + other.y, } } } impl std::ops::Sub for Point { type Output = Self; fn sub(self, other: Self) -> Self::Output { Self { x: self.x - other.x, y: self.y - other.y, } } } pub struct Polygon { points: Vec<Point>, } impl Polygon { pub fn new() -> Polygon { Polygon { points: Vec::new() } } pub fn add_point(&mut self, point: Point) { self.points.push(point); } pub fn left_most_point(&self) -> Option<Point> { self.points.iter().min_by_key(|p| p.x).copied() } pub fn iter(&self) -> impl Iterator<Item = &Point> { self.points.iter() } pub fn length(&self) -> f64 { if self.points.is_empty() { return 0.0; } let mut result = 0.0; let mut last_point = self.points[0]; for point in &self.points[1..] { result += last_point.dist(*point); last_point = *point; } result += last_point.dist(self.points[0]); result // Alternatively, Iterator::zip() lets us iterate over the points as pairs // but we need to pair each point with the next one, and the last point // with the first point. The zip() iterator is finished as soon as one of // the source iterators is finished, a neat trick is to combine Iterator::cycle // with Iterator::skip to create the second iterator for the zip and using map // and sum to calculate the total length. } } pub struct Circle { center: Point, radius: i32, } impl Circle { pub fn new(center: Point, radius: i32) -> Circle { Circle { center, radius } } pub fn circumference(&self) -> f64 { 2.0 * std::f64::consts::PI * f64::from(self.radius) } pub fn dist(&self, other: &Self) -> f64 { self.center.dist(other.center) } } pub enum Shape { Polygon(Polygon), Circle(Circle), } impl From<Polygon> for Shape { fn from(poly: Polygon) -> Self { Shape::Polygon(poly) } } impl From<Circle> for Shape { fn from(circle: Circle) -> Self { Shape::Circle(circle) } } impl Shape { pub fn perimeter(&self) -> f64 { match self { Shape::Polygon(poly) => poly.length(), Shape::Circle(circle) => circle.circumference(), } } } #[cfg(test)] mod tests { use super::*; fn round_two_digits(x: f64) -> f64 { (x * 100.0).round() / 100.0 } #[test] fn test_point_magnitude() { let p1 = Point::new(12, 13); assert_eq!(round_two_digits(p1.magnitude()), 17.69); } #[test] fn test_point_dist() { let p1 = Point::new(10, 10); let p2 = Point::new(14, 13); assert_eq!(round_two_digits(p1.dist(p2)), 5.00); } #[test] fn test_point_add() { let p1 = Point::new(16, 16); let p2 = p1 + Point::new(-4, 3); assert_eq!(p2, Point::new(12, 19)); } #[test] fn test_polygon_left_most_point() { let p1 = Point::new(12, 13); let p2 = Point::new(16, 16); let mut poly = Polygon::new(); poly.add_point(p1); poly.add_point(p2); assert_eq!(poly.left_most_point(), Some(p1)); } #[test] fn test_polygon_iter() { let p1 = Point::new(12, 13); let p2 = Point::new(16, 16); let mut poly = Polygon::new(); poly.add_point(p1); poly.add_point(p2); let points = poly.iter().cloned().collect::<Vec<_>>(); assert_eq!(points, vec![Point::new(12, 13), Point::new(16, 16)]); } #[test] fn test_shape_perimeters() { let mut poly = Polygon::new(); poly.add_point(Point::new(12, 13)); poly.add_point(Point::new(17, 11)); poly.add_point(Point::new(16, 16)); let shapes = vec![ Shape::from(poly), Shape::from(Circle::new(Point::new(10, 20), 5)), ]; let perimeters = shapes .iter() .map(Shape::perimeter) .map(round_two_digits) .collect::<Vec<_>>(); assert_eq!(perimeters, vec![15.48, 31.42]); } } fn main() {}
3์ผ์ฐจ ์คํ ์ฐ์ต๋ฌธ์
FFI๋ํผ
mod ffi { use std::os::raw::{c_char, c_int}; #[cfg(not(target_os = "macos"))] use std::os::raw::{c_long, c_ulong, c_ushort, c_uchar}; // Opaque type. See https://doc.rust-lang.org/nomicon/ffi.html. #[repr(C)] pub struct DIR { _data: [u8; 0], _marker: core::marker::PhantomData<(*mut u8, core::marker::PhantomPinned)>, } // Layout according to the Linux man page for readdir(3), where ino_t and // off_t are resolved according to the definitions in // /usr/include/x86_64-linux-gnu/{sys/types.h, bits/typesizes.h}. #[cfg(not(target_os = "macos"))] #[repr(C)] pub struct dirent { pub d_ino: c_ulong, pub d_off: c_long, pub d_reclen: c_ushort, pub d_type: c_uchar, pub d_name: [c_char; 256], } // Layout according to the macOS man page for dir(5). #[cfg(all(target_os = "macos"))] #[repr(C)] pub struct dirent { pub d_fileno: u64, pub d_seekoff: u64, pub d_reclen: u16, pub d_namlen: u16, pub d_type: u8, pub d_name: [c_char; 1024], } extern "C" { pub fn opendir(s: *const c_char) -> *mut DIR; #[cfg(not(all(target_os = "macos", target_arch = "x86_64")))] pub fn readdir(s: *mut DIR) -> *const dirent; // See https://github.com/rust-lang/libc/issues/414 and the section on // _DARWIN_FEATURE_64_BIT_INODE in the macOS man page for stat(2). // // "Platforms that existed before these updates were available" refers // to macOS (as opposed to iOS / wearOS / etc.) on Intel and PowerPC. #[cfg(all(target_os = "macos", target_arch = "x86_64"))] #[link_name = "readdir$INODE64"] pub fn readdir(s: *mut DIR) -> *const dirent; pub fn closedir(s: *mut DIR) -> c_int; } } use std::ffi::{CStr, CString, OsStr, OsString}; use std::os::unix::ffi::OsStrExt; #[derive(Debug)] struct DirectoryIterator { path: CString, dir: *mut ffi::DIR, } impl DirectoryIterator { fn new(path: &str) -> Result<DirectoryIterator, String> { // Call opendir and return a Ok value if that worked, // otherwise return Err with a message. let path = CString::new(path).map_err(|err| format!("Invalid path: {err}"))?; // SAFETY: path.as_ptr() cannot be NULL. let dir = unsafe { ffi::opendir(path.as_ptr()) }; if dir.is_null() { Err(format!("Could not open {:?}", path)) } else { Ok(DirectoryIterator { path, dir }) } } } impl Iterator for DirectoryIterator { type Item = OsString; fn next(&mut self) -> Option<OsString> { // Keep calling readdir until we get a NULL pointer back. // SAFETY: self.dir is never NULL. let dirent = unsafe { ffi::readdir(self.dir) }; if dirent.is_null() { // We have reached the end of the directory. return None; } // SAFETY: dirent is not NULL and dirent.d_name is NUL // terminated. let d_name = unsafe { CStr::from_ptr((*dirent).d_name.as_ptr()) }; let os_str = OsStr::from_bytes(d_name.to_bytes()); Some(os_str.to_owned()) } } impl Drop for DirectoryIterator { fn drop(&mut self) { // Call closedir as needed. if !self.dir.is_null() { // SAFETY: self.dir is not NULL. if unsafe { ffi::closedir(self.dir) } != 0 { panic!("Could not close {:?}", self.path); } } } } fn main() -> Result<(), String> { let iter = DirectoryIterator::new(".")?; println!("files: {:#?}", iter.collect::<Vec<_>>()); Ok(()) } #[cfg(test)] mod tests { use super::*; use std::error::Error; #[test] fn test_nonexisting_directory() { let iter = DirectoryIterator::new("no-such-directory"); assert!(iter.is_err()); } #[test] fn test_empty_directory() -> Result<(), Box<dyn Error>> { let tmp = tempfile::TempDir::new()?; let iter = DirectoryIterator::new( tmp.path().to_str().ok_or("Non UTF-8 character in path")?, )?; let mut entries = iter.collect::<Vec<_>>(); entries.sort(); assert_eq!(entries, &[".", ".."]); Ok(()) } #[test] fn test_nonempty_directory() -> Result<(), Box<dyn Error>> { let tmp = tempfile::TempDir::new()?; std::fs::write(tmp.path().join("foo.txt"), "The Foo Diaries\n")?; std::fs::write(tmp.path().join("bar.png"), "<PNG>\n")?; std::fs::write(tmp.path().join("crab.rs"), "//! Crab\n")?; let iter = DirectoryIterator::new( tmp.path().to_str().ok_or("Non UTF-8 character in path")?, )?; let mut entries = iter.collect::<Vec<_>>(); entries.sort(); assert_eq!(entries, &[".", "..", "bar.png", "crab.rs", "foo.txt"]); Ok(()) } }
Bare Metal Rust Morning Exercise
๋์นจ๋ฐ
#![no_main] #![no_std] extern crate panic_halt as _; use core::fmt::Write; use cortex_m_rt::entry; use core::cmp::{max, min}; use lsm303agr::{AccelOutputDataRate, Lsm303agr, MagOutputDataRate}; use microbit::display::blocking::Display; use microbit::hal::prelude::*; use microbit::hal::twim::Twim; use microbit::hal::uarte::{Baudrate, Parity, Uarte}; use microbit::hal::Timer; use microbit::pac::twim0::frequency::FREQUENCY_A; use microbit::Board; const COMPASS_SCALE: i32 = 30000; const ACCELEROMETER_SCALE: i32 = 700; #[entry] fn main() -> ! { let board = Board::take().unwrap(); // Configure serial port. let mut serial = Uarte::new( board.UARTE0, board.uart.into(), Parity::EXCLUDED, Baudrate::BAUD115200, ); // Set up the I2C controller and Inertial Measurement Unit. writeln!(serial, "Setting up IMU...").unwrap(); let i2c = Twim::new(board.TWIM0, board.i2c_internal.into(), FREQUENCY_A::K100); let mut imu = Lsm303agr::new_with_i2c(i2c); imu.init().unwrap(); imu.set_mag_odr(MagOutputDataRate::Hz50).unwrap(); imu.set_accel_odr(AccelOutputDataRate::Hz50).unwrap(); let mut imu = imu.into_mag_continuous().ok().unwrap(); // Set up display and timer. let mut timer = Timer::new(board.TIMER0); let mut display = Display::new(board.display_pins); let mut mode = Mode::Compass; let mut button_pressed = false; writeln!(serial, "Ready.").unwrap(); loop { // Read compass data and log it to the serial port. while !(imu.mag_status().unwrap().xyz_new_data && imu.accel_status().unwrap().xyz_new_data) {} let compass_reading = imu.mag_data().unwrap(); let accelerometer_reading = imu.accel_data().unwrap(); writeln!( serial, "{},{},{}\t{},{},{}", compass_reading.x, compass_reading.y, compass_reading.z, accelerometer_reading.x, accelerometer_reading.y, accelerometer_reading.z, ) .unwrap(); let mut image = [[0; 5]; 5]; let (x, y) = match mode { Mode::Compass => ( scale(-compass_reading.x, -COMPASS_SCALE, COMPASS_SCALE, 0, 4) as usize, scale(compass_reading.y, -COMPASS_SCALE, COMPASS_SCALE, 0, 4) as usize, ), Mode::Accelerometer => ( scale( accelerometer_reading.x, -ACCELEROMETER_SCALE, ACCELEROMETER_SCALE, 0, 4, ) as usize, scale( -accelerometer_reading.y, -ACCELEROMETER_SCALE, ACCELEROMETER_SCALE, 0, 4, ) as usize, ), }; image[y][x] = 255; display.show(&mut timer, image, 100); // If button A is pressed, switch to the next mode and briefly blink all LEDs on. if board.buttons.button_a.is_low().unwrap() { if !button_pressed { mode = mode.next(); display.show(&mut timer, [[255; 5]; 5], 200); } button_pressed = true; } else { button_pressed = false; } } } #[derive(Copy, Clone, Debug, Eq, PartialEq)] enum Mode { Compass, Accelerometer, } impl Mode { fn next(self) -> Self { match self { Self::Compass => Self::Accelerometer, Self::Accelerometer => Self::Compass, } } } fn scale(value: i32, min_in: i32, max_in: i32, min_out: i32, max_out: i32) -> i32 { let range_in = max_in - min_in; let range_out = max_out - min_out; cap( min_out + range_out * (value - min_in) / range_in, min_out, max_out, ) } fn cap(value: i32, min_value: i32, max_value: i32) -> i32 { max(min_value, min(value, max_value)) }
์ Bare Metal ์คํ
RTC driver
main.rs
:
#![no_main] #![no_std] mod exceptions; mod logger; mod pl011; mod pl031; use crate::pl031::Rtc; use arm_gic::gicv3::{IntId, Trigger}; use arm_gic::{irq_enable, wfi}; use chrono::{TimeZone, Utc}; use core::hint::spin_loop; use crate::pl011::Uart; use arm_gic::gicv3::GicV3; use core::panic::PanicInfo; use log::{error, info, trace, LevelFilter}; use smccc::psci::system_off; use smccc::Hvc; /// Base addresses of the GICv3. const GICD_BASE_ADDRESS: *mut u64 = 0x800_0000 as _; const GICR_BASE_ADDRESS: *mut u64 = 0x80A_0000 as _; /// Base address of the primary PL011 UART. const PL011_BASE_ADDRESS: *mut u32 = 0x900_0000 as _; /// Base address of the PL031 RTC. const PL031_BASE_ADDRESS: *mut u32 = 0x901_0000 as _; /// The IRQ used by the PL031 RTC. const PL031_IRQ: IntId = IntId::spi(2); #[no_mangle] extern "C" fn main(x0: u64, x1: u64, x2: u64, x3: u64) { // Safe because `PL011_BASE_ADDRESS` is the base address of a PL011 device, // and nothing else accesses that address range. let uart = unsafe { Uart::new(PL011_BASE_ADDRESS) }; logger::init(uart, LevelFilter::Trace).unwrap(); info!("main({:#x}, {:#x}, {:#x}, {:#x})", x0, x1, x2, x3); // Safe because `GICD_BASE_ADDRESS` and `GICR_BASE_ADDRESS` are the base // addresses of a GICv3 distributor and redistributor respectively, and // nothing else accesses those address ranges. let mut gic = unsafe { GicV3::new(GICD_BASE_ADDRESS, GICR_BASE_ADDRESS) }; gic.setup(); // Safe because `PL031_BASE_ADDRESS` is the base address of a PL031 device, // and nothing else accesses that address range. let mut rtc = unsafe { Rtc::new(PL031_BASE_ADDRESS) }; let timestamp = rtc.read(); let time = Utc.timestamp_opt(timestamp.into(), 0).unwrap(); info!("RTC: {time}"); GicV3::set_priority_mask(0xff); gic.set_interrupt_priority(PL031_IRQ, 0x80); gic.set_trigger(PL031_IRQ, Trigger::Level); irq_enable(); gic.enable_interrupt(PL031_IRQ, true); // Wait for 3 seconds, without interrupts. let target = timestamp + 3; rtc.set_match(target); info!( "Waiting for {}", Utc.timestamp_opt(target.into(), 0).unwrap() ); trace!( "matched={}, interrupt_pending={}", rtc.matched(), rtc.interrupt_pending() ); while !rtc.matched() { spin_loop(); } trace!( "matched={}, interrupt_pending={}", rtc.matched(), rtc.interrupt_pending() ); info!("Finished waiting"); // Wait another 3 seconds for an interrupt. let target = timestamp + 6; info!( "Waiting for {}", Utc.timestamp_opt(target.into(), 0).unwrap() ); rtc.set_match(target); rtc.clear_interrupt(); rtc.enable_interrupt(true); trace!( "matched={}, interrupt_pending={}", rtc.matched(), rtc.interrupt_pending() ); while !rtc.interrupt_pending() { wfi(); } trace!( "matched={}, interrupt_pending={}", rtc.matched(), rtc.interrupt_pending() ); info!("Finished waiting"); system_off::<Hvc>().unwrap(); } #[panic_handler] fn panic(info: &PanicInfo) -> ! { error!("{info}"); system_off::<Hvc>().unwrap(); loop {} }
pl031.rs
:
#![allow(unused)] fn main() { use core::ptr::{addr_of, addr_of_mut}; #[repr(C, align(4))] struct Registers { /// Data register dr: u32, /// Match register mr: u32, /// Load register lr: u32, /// Control register cr: u8, _reserved0: [u8; 3], /// Interrupt Mask Set or Clear register imsc: u8, _reserved1: [u8; 3], /// Raw Interrupt Status ris: u8, _reserved2: [u8; 3], /// Masked Interrupt Status mis: u8, _reserved3: [u8; 3], /// Interrupt Clear Register icr: u8, _reserved4: [u8; 3], } /// Driver for a PL031 real-time clock. #[derive(Debug)] pub struct Rtc { registers: *mut Registers, } impl Rtc { /// Constructs a new instance of the RTC driver for a PL031 device at the /// given base address. /// /// # Safety /// /// The given base address must point to the MMIO control registers of a /// PL031 device, which must be mapped into the address space of the process /// as device memory and not have any other aliases. pub unsafe fn new(base_address: *mut u32) -> Self { Self { registers: base_address as *mut Registers, } } /// Reads the current RTC value. pub fn read(&self) -> u32 { // Safe because we know that self.registers points to the control // registers of a PL031 device which is appropriately mapped. unsafe { addr_of!((*self.registers).dr).read_volatile() } } /// Writes a match value. When the RTC value matches this then an interrupt /// will be generated (if it is enabled). pub fn set_match(&mut self, value: u32) { // Safe because we know that self.registers points to the control // registers of a PL031 device which is appropriately mapped. unsafe { addr_of_mut!((*self.registers).mr).write_volatile(value) } } /// Returns whether the match register matches the RTC value, whether or not /// the interrupt is enabled. pub fn matched(&self) -> bool { // Safe because we know that self.registers points to the control // registers of a PL031 device which is appropriately mapped. let ris = unsafe { addr_of!((*self.registers).ris).read_volatile() }; (ris & 0x01) != 0 } /// Returns whether there is currently an interrupt pending. /// /// This should be true if and only if `matched` returns true and the /// interrupt is masked. pub fn interrupt_pending(&self) -> bool { // Safe because we know that self.registers points to the control // registers of a PL031 device which is appropriately mapped. let ris = unsafe { addr_of!((*self.registers).mis).read_volatile() }; (ris & 0x01) != 0 } /// Sets or clears the interrupt mask. /// /// When the mask is true the interrupt is enabled; when it is false the /// interrupt is disabled. pub fn enable_interrupt(&mut self, mask: bool) { let imsc = if mask { 0x01 } else { 0x00 }; // Safe because we know that self.registers points to the control // registers of a PL031 device which is appropriately mapped. unsafe { addr_of_mut!((*self.registers).imsc).write_volatile(imsc) } } /// Clears a pending interrupt, if any. pub fn clear_interrupt(&mut self) { // Safe because we know that self.registers points to the control // registers of a PL031 device which is appropriately mapped. unsafe { addr_of_mut!((*self.registers).icr).write_volatile(0x01) } } } // Safe because it just contains a pointer to device memory, which can be // accessed from any context. unsafe impl Send for Rtc {} }
Concurrency Morning Exercise
์์ฌํ๋ ์ฒ ํ์๋ค
use std::sync::{mpsc, Arc, Mutex}; use std::thread; use std::time::Duration; struct Fork; struct Philosopher { name: String, left_fork: Arc<Mutex<Fork>>, right_fork: Arc<Mutex<Fork>>, thoughts: mpsc::SyncSender<String>, } impl Philosopher { fn think(&self) { self.thoughts .send(format!("Eureka! {} has a new idea!", &self.name)) .unwrap(); } fn eat(&self) { println!("{} is trying to eat", &self.name); let left = self.left_fork.lock().unwrap(); let right = self.right_fork.lock().unwrap(); println!("{} is eating...", &self.name); thread::sleep(Duration::from_millis(10)); } } static PHILOSOPHERS: &[&str] = &["Socrates", "Plato", "Aristotle", "Thales", "Pythagoras"]; fn main() { let (tx, rx) = mpsc::sync_channel(10); let forks = (0..PHILOSOPHERS.len()) .map(|_| Arc::new(Mutex::new(Fork))) .collect::<Vec<_>>(); for i in 0..forks.len() { let tx = tx.clone(); let mut left_fork = Arc::clone(&forks[i]); let mut right_fork = Arc::clone(&forks[(i + 1) % forks.len()]); // To avoid a deadlock, we have to break the symmetry // somewhere. This will swap the forks without deinitializing // either of them. if i == forks.len() - 1 { std::mem::swap(&mut left_fork, &mut right_fork); } let philosopher = Philosopher { name: PHILOSOPHERS[i].to_string(), thoughts: tx, left_fork, right_fork, }; thread::spawn(move || { for _ in 0..100 { philosopher.eat(); philosopher.think(); } }); } drop(tx); for thought in rx { println!("{thought}"); } }
Link Checker
use std::{sync::Arc, sync::Mutex, sync::mpsc, thread}; use reqwest::{blocking::Client, Url}; use scraper::{Html, Selector}; use thiserror::Error; #[derive(Error, Debug)] enum Error { #[error("request error: {0}")] ReqwestError(#[from] reqwest::Error), #[error("bad http response: {0}")] BadResponse(String), } #[derive(Debug)] struct CrawlCommand { url: Url, extract_links: bool, } fn visit_page(client: &Client, command: &CrawlCommand) -> Result<Vec<Url>, Error> { println!("Checking {:#}", command.url); let response = client.get(command.url.clone()).send()?; if !response.status().is_success() { return Err(Error::BadResponse(response.status().to_string())); } let mut link_urls = Vec::new(); if !command.extract_links { return Ok(link_urls); } let base_url = response.url().to_owned(); let body_text = response.text()?; let document = Html::parse_document(&body_text); let selector = Selector::parse("a").unwrap(); let href_values = document .select(&selector) .filter_map(|element| element.value().attr("href")); for href in href_values { match base_url.join(href) { Ok(link_url) => { link_urls.push(link_url); } Err(err) => { println!("On {base_url:#}: ignored unparsable {href:?}: {err}"); } } } Ok(link_urls) } struct CrawlState { domain: String, visited_pages: std::collections::HashSet<String>, } impl CrawlState { fn new(start_url: &Url) -> CrawlState { let mut visited_pages = std::collections::HashSet::new(); visited_pages.insert(start_url.as_str().to_string()); CrawlState { domain: start_url.domain().unwrap().to_string(), visited_pages, } } /// Determine whether links within the given page should be extracted. fn should_extract_links(&self, url: &Url) -> bool { let Some(url_domain) = url.domain() else { return false; }; url_domain == self.domain } /// Mark the given page as visited, returning true if it had already /// been visited. fn mark_visited(&mut self, url: &Url) -> bool { self.visited_pages.insert(url.as_str().to_string()) } } type CrawlResult = Result<Vec<Url>, (Url, Error)>; fn spawn_crawler_threads( command_receiver: mpsc::Receiver<CrawlCommand>, result_sender: mpsc::Sender<CrawlResult>, thread_count: u32, ) { let command_receiver = Arc::new(Mutex::new(command_receiver)); for _ in 0..thread_count { let result_sender = result_sender.clone(); let command_receiver = command_receiver.clone(); thread::spawn(move || { let client = Client::new(); loop { let command_result = { let receiver_guard = command_receiver.lock().unwrap(); receiver_guard.recv() }; let Ok(crawl_command) = command_result else { // The sender got dropped. No more commands coming in. break; }; let crawl_result = match visit_page(&client, &crawl_command) { Ok(link_urls) => Ok(link_urls), Err(error) => Err((crawl_command.url, error)), }; result_sender.send(crawl_result).unwrap(); } }); } } fn control_crawl( start_url: Url, command_sender: mpsc::Sender<CrawlCommand>, result_receiver: mpsc::Receiver<CrawlResult>, ) -> Vec<Url> { let mut crawl_state = CrawlState::new(&start_url); let start_command = CrawlCommand { url: start_url, extract_links: true }; command_sender.send(start_command).unwrap(); let mut pending_urls = 1; let mut bad_urls = Vec::new(); while pending_urls > 0 { let crawl_result = result_receiver.recv().unwrap(); pending_urls -= 1; match crawl_result { Ok(link_urls) => { for url in link_urls { if crawl_state.mark_visited(&url) { let extract_links = crawl_state.should_extract_links(&url); let crawl_command = CrawlCommand { url, extract_links }; command_sender.send(crawl_command).unwrap(); pending_urls += 1; } } } Err((url, error)) => { bad_urls.push(url); println!("Got crawling error: {:#}", error); continue; } } } bad_urls } fn check_links(start_url: Url) -> Vec<Url> { let (result_sender, result_receiver) = mpsc::channel::<CrawlResult>(); let (command_sender, command_receiver) = mpsc::channel::<CrawlCommand>(); spawn_crawler_threads(command_receiver, result_sender, 16); control_crawl(start_url, command_sender, result_receiver) } fn main() { let start_url = reqwest::Url::parse("https://www.google.org").unwrap(); let bad_urls = check_links(start_url); println!("Bad URLs: {:#?}", bad_urls); }
Concurrency Afternoon Exercise
Dining Philosophers - Async
use std::sync::Arc; use tokio::time; use tokio::sync::mpsc::{self, Sender}; use tokio::sync::Mutex; struct Fork; struct Philosopher { name: String, left_fork: Arc<Mutex<Fork>>, right_fork: Arc<Mutex<Fork>>, thoughts: Sender<String>, } impl Philosopher { async fn think(&self) { self.thoughts .send(format!("Eureka! {} has a new idea!", &self.name)).await .unwrap(); } async fn eat(&self) { // Pick up forks... let _first_lock = self.left_fork.lock().await; // Add a delay before picking the second fork to allow the execution // to transfer to another task time::sleep(time::Duration::from_millis(1)).await; let _second_lock = self.right_fork.lock().await; println!("{} is eating...", &self.name); time::sleep(time::Duration::from_millis(5)).await; // The locks are dropped here } } static PHILOSOPHERS: &[&str] = &["Socrates", "Plato", "Aristotle", "Thales", "Pythagoras"]; #[tokio::main] async fn main() { // Create forks let mut forks = vec![]; (0..PHILOSOPHERS.len()).for_each(|_| forks.push(Arc::new(Mutex::new(Fork)))); // Create philosophers let (philosophers, mut rx) = { let mut philosophers = vec![]; let (tx, rx) = mpsc::channel(10); for (i, name) in PHILOSOPHERS.iter().enumerate() { let left_fork = Arc::clone(&forks[i]); let right_fork = Arc::clone(&forks[(i + 1) % PHILOSOPHERS.len()]); // To avoid a deadlock, we have to break the symmetry // somewhere. This will swap the forks without deinitializing // either of them. if i == 0 { std::mem::swap(&mut left_fork, &mut right_fork); } philosophers.push(Philosopher { name: name.to_string(), left_fork, right_fork, thoughts: tx.clone(), }); } (philosophers, rx) // tx is dropped here, so we don't need to explicitly drop it later }; // Make them think and eat for phil in philosophers { tokio::spawn(async move { for _ in 0..100 { phil.think().await; phil.eat().await; } }); } // Output their thoughts while let Some(thought) = rx.recv().await { println!("Here is a thought: {thought}"); } }
์ฑํ ์ ํ๋ฆฌ์ผ์ด์
src/bin/server.rs
:
use futures_util::sink::SinkExt; use futures_util::stream::StreamExt; use std::error::Error; use std::net::SocketAddr; use tokio::net::{TcpListener, TcpStream}; use tokio::sync::broadcast::{channel, Sender}; use tokio_websockets::{Message, ServerBuilder, WebsocketStream}; async fn handle_connection( addr: SocketAddr, mut ws_stream: WebsocketStream<TcpStream>, bcast_tx: Sender<String>, ) -> Result<(), Box<dyn Error + Send + Sync>> { ws_stream .send(Message::text("Welcome to chat! Type a message".into())) .await?; let mut bcast_rx = bcast_tx.subscribe(); // A continuous loop for concurrently performing two tasks: (1) receiving // messages from `ws_stream` and broadcasting them, and (2) receiving // messages on `bcast_rx` and sending them to the client. loop { tokio::select! { incoming = ws_stream.next() => { match incoming { Some(Ok(msg)) => { if let Some(text) = msg.as_text() { println!("From client {addr:?} {text:?}"); bcast_tx.send(text.into())?; } } Some(Err(err)) => return Err(err.into()), None => return Ok(()), } } msg = bcast_rx.recv() => { ws_stream.send(Message::text(msg?)).await?; } } } } #[tokio::main] async fn main() -> Result<(), Box<dyn Error + Send + Sync>> { let (bcast_tx, _) = channel(16); let listener = TcpListener::bind("127.0.0.1:2000").await?; println!("listening on port 2000"); loop { let (socket, addr) = listener.accept().await?; println!("New connection from {addr:?}"); let bcast_tx = bcast_tx.clone(); tokio::spawn(async move { // Wrap the raw TCP stream into a websocket. let ws_stream = ServerBuilder::new().accept(socket).await?; handle_connection(addr, ws_stream, bcast_tx).await }); } }
src/bin/client.rs
:
use futures_util::stream::StreamExt; use futures_util::SinkExt; use http::Uri; use tokio::io::{AsyncBufReadExt, BufReader}; use tokio_websockets::{ClientBuilder, Message}; #[tokio::main] async fn main() -> Result<(), tokio_websockets::Error> { let (mut ws_stream, _) = ClientBuilder::from_uri(Uri::from_static("ws://127.0.0.1:2000")) .connect() .await?; let stdin = tokio::io::stdin(); let mut stdin = BufReader::new(stdin).lines(); // Continuous loop for concurrently sending and receiving messages. loop { tokio::select! { incoming = ws_stream.next() => { match incoming { Some(Ok(msg)) => { if let Some(text) = msg.as_text() { println!("From server: {}", text); } }, Some(Err(err)) => return Err(err.into()), None => return Ok(()), } } res = stdin.next_line() => { match res { Ok(None) => return Ok(()), Ok(Some(line)) => ws_stream.send(Message::text(line.to_string())).await?, Err(err) => return Err(err.into()), } } } } }