STUDY/Rust
Rust - 18. 스마트 포인터 (1) Box<T>, Deref, Drop
sinawi95
2024. 3. 21. 21:12
728x90
또 포인터야? 오늘도 포인터 때문에 이해안돼서 애먹었는데..?
포인터(pointer): 메모리의 주솟값을 담고 있는 변수에 대한 일반적인 개념
스마트 포인터(smart pointer): 포인터 + 추가적인 메타데이터와 능력
- C++에서 유래됨. 이해하기 어려우면 C++에서 먼저 이해하고 오자
1. Box<T>
Box<T>
- 힙에 데이터 저장
- 스택과 비교해서 힙에 저장하는것 제외하면 성능 측면에서 오버헤드 없음
사용하는 이유(이해 못함)
- 컴파일 타임에는 크기를 알 수 없는 타입이 있는데, 정확한 크기를 요구하는 컨텍스트 내에서 그 타입의 값을 사용하고 싶을 때
- 재귀적 타입을 사용하면 공간의 크기를 알기 어려움.
- Box<T>인 포인터로 대체하면 크기가 고정되므로 사용할수 있음
- 커다란 데이터를 가지고 있고 소유권을 옮기고 싶지만 그렇게 했을 때 데이터가 복사되지 않을 것을 보장하고 싶을 때
- 스택에 포인터 데이터만 복사하고 다량의 데이터는 힙에 두고 사용
- 어떤 값을 소유하고 이 값의 구체화된 타입보다는 특정 트레이트를 구현한 타입이라는 점만 신경 쓰고 싶을 때
- 트레이트 객체
예시
fn main() {
let b = Box::new(5);
println!("b = {}", b);
}
enum List {
Cons(i32, Box<List>),
Nil,
}
use crate::List::{Cons, Nil};
fn main() {
let list = Cons(1, Box::new(Cons(2, Box::new(Cons(3, Box::new(Nil))))));
}
2. Deref 트레이트, 스마트 포인터를 보통참조자처럼 취급하기
Deref 트레이트
- 역참조 연산자 *(dereference operator) 동작을 커스터마이징 할 수 있음.
use std::ops::Deref;
impl<T> Deref for MyBox<T> {
type Target = T;
fn deref(&self) -> &Self::Target {
&self.0
}
}
struct MyBox<T>(T);
impl<T> MyBox<T> {
fn new(x: T) -> MyBox<T> {
MyBox(x)
}
}
fn main() {
let x = 5;
let y = MyBox::new(x);
assert_eq!(5, x);
assert_eq!(5, *y); //역참조
}
Deref coercion 역참조 강제
- 어떤 타입의 참조자를 다른 타입의 참조자로 변경
- 명시적인 참조, 역참조를 추가할 필요가 없도록 하기위해 도입됨
역참조 강제가 가변성과 상호작용하는 법
- T: Deref<Target=U>일 때 &T에서 &U로
- T: DerefMut<Target=U>일 때 &mut T에서 &mut U로
- T: Deref<Target=U>일 때 &mut T에서 &U로
3. Drop 트레이트
Drop: 메모리 정리 코드 실행
- 값이 스코프 밖으로 벗어날때마다 실행되는 코드
- 버려지는 순서는 만들어진 순서의 역순으로 진행됨
std::mem::drop
- Drop 트레이트가 실행되기 전에 먼저 정리 하고 싶은 경우 사용
- 두 번 정리하는 것을 막음
https://doc.rust-kr.org/ch15-00-smart-pointers.html