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'
๋ฐํ์ ํจ๋์ด ๋ฐ์ํจ์ ๋ณด์ด์ธ์.