반응형
스마트 포인터(smart pointer)
- C++ 프로그램에서 new 키워드를 사용하여 동적으로 할당받은 메모리는 반드시 delete 키워드를 사용하여 해제해야 합니다.
- C++에서는 메모리 누수(memory leak)로부터 프로그램의 안정성을 보장하기 위해 스마트 포인터를 제공하고 있습니다.
- 스마트 포인터란 포인터처럼 동작하는 클래스 템플릿으로 사용이 끝난 메모리를 자동으로 해제해 줍니다.
스마트 포인터의 동작
- 보통 new 키워드를 사용해 기본 포인터(raw pointer)가 실제 메모리를 가리키도록 초기화한 후에 기본 포인터를 스마트 포인터에 대입하여 사용합니다.
- 이렇게 정의된 스마트 포인터의 수명이 다하면 소멸자는 delete 키워드를 사용하여 할당된 메모리를 자동으로 해제합니다.
- 따라서 new 키워드가 반환하는 주소값을 스마트 포인터에 대입하면 따로 메모리를 해제할 필요가 없어집니다.
스마트 포인터의 종류
- C++11 이전에는 auto_ptr이라는 스마트 포인터를 사용하여 이 작업을 수행하였습니다.
- 하지만 C++11부터는 다음과 같은 새로운 스마트 포인터를 제공하고 있습니다.
- unique_ptr
- shared_ptr
- weak_ptr
- 스마트 포인터는 memory 헤더 파일에 정의되어 있습니다.
※auto_ptr은 C++11 표준부터 삭제되었습니다.
unique_ptr
- 하나의 스마트 포인터만이 특정 객체를 소유할 수 있도록, 객체에 소유권 개념을 도입한 스마트 포인터입니다.
- 해당 객체의 소유권을 가지고 있을 때만, 소멸자가 해당 객체를 삭제할 수 있습니다.
- move() 멤버 함수를 통해 소유권을 이전할 수는 있지만, 복사할 수는 없습니다.
- 소유권이 이전되면, 이전 unique_ptr 인스턴스는 더는 해당 객체를 소유하지 않게 재설정됩니다.
- 보통의 C++ 객체에 대해 스마트 포인터가 필요한 상황에서는 주로 unique_ptr을 사용하면 됩니다.
- C++14 이후부터 제공되는 make_unique() 함수를 사용하면 unique_ptr 인스턴스를 안전하게 생성할 수 있습니다.
- make_unique() 함수는 전달받은 인수를 사용해 지정된 타입의 객체를 생성하고, 생성된 객체를 가리키는 unique_ptr을 반환합니다.
- 이 함수를 사용하면, 예외 발생에 대해 안전하게 대처할 수 있습니다.
- make_unique() 함수를 사용하기 위해서는 여러분의 C++ 컴파일러가 C++14를 지원해야 합니다.
shared_ptr
- 하나의 특정 객체를 참조하는 스마트 포인터가 총 몇 개인지를 참조하는 스마트 포인터입니다.
- 이렇게 참조하고 있는 스마트 포인터의 개수를 참조 횟수(reference count)라고 합니다.
- 참조 횟수는 특정 객체에 새로운 shared_ptr이 추가될 때마다 1씩 증가하며, 수명이 다할 때마다 1씩 감소합니다.
- 따라서 마지막 shared_ptr의 수명이 다하여, 참조 횟수가 0이 되면 delete 키워드를 사용하여 메모리를 자동으로 해제합니다.
weak_ptr
- 하나 이상의 shared_ptr 인스턴스가 소유하는 객체에 대한 접근을 제공하지만, 소유자의 수에는 포함되지 않는 스마트 포인터입니다.
- shared_ptr은 참조 횟수(reference count)를 기반으로 동작하는 스마트 포인터입니다.
- 만약 서로가 상대방을 가리키는 shared_ptr를 가지고 있다면, 참조 횟수는 절대 0이 되지 않으므로 메모리는 영원히 해제되지 않습니다.
- 이렇게 서로가 상대방을 참조하고 있는 상황을 순환 참조(circular reference)라고 합니다.
- weak_ptr은 바로 이러한 shared_ptr 인스턴스 사이의 순환 참조를 제거하기 위해서 사용됩니다.
참고
http://tcpschool.com/cpp/cpp_template_smartPointer
https://docs.microsoft.com/ko-kr/cpp/cpp/smart-pointers-modern-cpp?view=msvc-170
반응형
'C, C++' 카테고리의 다른 글
[C#] ThreadPool 클래스 (0) | 2022.07.04 |
---|---|
[C/C++]explicit (0) | 2021.12.01 |
[C/C++] accumulate 사용법과 주의사항 (0) | 2021.07.09 |
[C/C++] stringstream 사용법 (0) | 2021.07.03 |
[C/C++]enum과 enum class (0) | 2021.04.05 |