C++ åèæå
- C++11
- C++14
- C++17
- C++20
- C++ ç¼è¯å¨æ¯ææ åµè¡¨
- ç¬ç«ä¸å®¿ä¸»å®ç°
- C++ è¯è¨
- C++ å ³é®è¯
- é¢å¤çå¨
- C++ æ ååºå¤´æä»¶
- å ·åè¦æ±
- åè½ç¹æ§æµè¯ (C++20)
- å·¥å ·åº
- ç±»åæ¯æï¼åºæ¬ç±»åãRTTIãç±»åç¹æ§ï¼
- æ¦å¿µåº (C++20)
- é误å¤ç
- 卿å å管ç
- std::unique_ptr
- std::make_unique, std::make_unique_for_overwrite
- std::hash <std::unique_ptr>
- std::unique_ptr<T,Deleter>::operator<<
- std::swap(std::unique_ptr)
- std::unique_ptr<T,Deleter>::operator*
- std::unique_ptr<T,Deleter>::operator[]
- operator==,!=,<,<=,>,>=,<=>(std::unique_ptr)
- std::unique_ptr<T,Deleter>::get
- std::unique_ptr<T,Deleter>::get_deleter
- std::unique_ptr<T,Deleter>::operator bool
- std::unique_ptr<T,Deleter>::reset
- std::unique_ptr<T,Deleter>::swap
- std::unique_ptr<T,Deleter>::operator=
- std::unique_ptr<T,Deleter>::release
- std::unique_ptr<T,Deleter>::unique_ptr
- std::unique_ptr<T,Deleter>::~unique_ptr
- std::addressof
- std::allocator_traits
- std::default_delete
- std::allocator_arg_t
- std::allocator_arg
- std::weak_ptr
- std::enable_shared_from_this
- std::bad_weak_ptr
- std::scoped_allocator_adaptor
- std::auto_ptr
- std::destroy_at
- std::destroy
- std::destroy_n
- std::uninitialized_move
- std::uninitialized_value_construct
- std::owner_less
- std::shared_ptr
- std::to_address
- std::assume_aligned
- std::make_obj_using_allocator
- C å å管çåº
- ä½å±å å管ç
- std::pmr::memory_resource
- std::allocator
- std::pointer_traits
- std::uses_allocator
- std::uses_allocator_construction_args
- std::uninitialized_construct_using_allocator
- std::pmr::polymorphic_allocator
- std::pmr::get_default_resource
- std::pmr::set_default_resource
- std::pmr::new_delete_resource
- std::pmr::null_memory_resource
- std::pmr::synchronized_pool_resource
- std::pmr::unsynchronized_pool_resource
- std::pmr::monotonic_buffer_resource
- std::pmr::pool_options
- std::raw_storage_iterator
- std::get_temporary_buffer
- std::return_temporary_buffer
- std::uninitialized_copy
- std::uninitialized_fill
- std::uninitialized_default_construct
- std::uninitialized_copy_n
- std::uninitialized_fill_n
- std::uninitialized_move_n
- std::uninitialized_default_construct_n
- std::uninitialized_value_construct_n
- std::construct_at
- std::align
- 注é
- æ¥æåæ¶é´å·¥å ·
- å符串åº
- 容å¨åº
- è¿ä»£å¨åº
- èå´åº (C++20)
- ç®æ³åº
- æ°å¼åº
- è¾å ¥/è¾åºåº
- æä»¶ç³»ç»åº
- æ¬å°ååº
- æ£å表达å¼åº
- ååæä½åº
- çº¿ç¨æ¯æåº
- å®éªæ§ C++ ç¹æ§
- æç¨çèµæº
- ç´¢å¼
- std 符å·ç´¢å¼
- åç¨æ¯æ (C++20)
- C++ å ³é®è¯
ä½ç½®ï¼é¦é¡µ > C++ åèæå >卿å å管ç > std::unique_ptr
std::unique_ptr
  class T,
  class Deleter
std::unique_ptr æ¯éè¿æéå æå¹¶ç®¡çå¦ä¸å¯¹è±¡ï¼å¹¶å¨ unique_ptr 离å¼ä½ç¨åæ¶éæ¾è¯¥å¯¹è±¡çæºè½æéã
å¨ä¸å两è ä¹ä¸åçæ¶ç¨å ³èçå é¤å¨éæ¾å¯¹è±¡ï¼
- 鿝äºç®¡çç
unique_ptr对象 - éè¿ operator= æ reset() èµå¼å¦ä¸æéç»ç®¡çç
unique_ptr对象ã
éè¿è°ç¨ get_deleter()(ptr) ï¼ç¨æ½å¨ä¸ºç¨æ·æä¾çå é¤å¨éæ¾å¯¹è±¡ãé»è®¤å é¤å¨ç¨ delete è¿ç®ç¬¦ï¼å®éæ¯å¯¹è±¡å¹¶è§£åé å åã
unique_ptr 亦å¯ä»¥ä¸å æå¯¹è±¡ï¼è¯¥æ
åµä¸ç§°å®ä¸ºç©º (empty)ã
std::unique_ptr æä¸¤ä¸ªçæ¬ï¼
类满足å¯ç§»å¨æé (MoveConstructible) åå¯ç§»å¨èµå¼ (MoveAssignable) çè¦æ±ï¼ä½ä¸æ»¡è¶³å¯å¤å¶æé (CopyConstructible) æå¯å¤å¶èµå¼ (CopyAssignable) çè¦æ±ã
| ç±»åè¦æ± | ||
-Deleter å¿
é¡»æ¯å½æ°å¯¹è±¡ (FunctionObject) æå°å½æ°å¯¹è±¡ (FunctionObject) çå·¦å¼å¼ç¨æå°å½æ°çå·¦å¼å¼ç¨ï¼å¯ä»¥ unique_ptr<T, Deleter>::pointer ç±»ååæ°è°ç¨
|
注解
åªæé const ç unique_ptr è½è½¬ç§»è¢«ç®¡çå¯¹è±¡çæææç»å¦ä¸ unique_ptr ãè¥å¯¹è±¡ççåæä¸º const std::unique_ptr æç®¡çï¼åå®è¢«éå®å¨å建æéçä½ç¨åä¸ã
std::unique_ptr 常ç¨äºç®¡ç对象ççåæï¼å
å«ï¼
- éè¿æ£å¸¸éåºåç»ç±å¼å¸¸éåºä¸¤è ä¸çåä¿è¯å é¤ï¼æä¾å¼å¸¸å®å ¨ï¼ç»å¤çæ¥æå¨æçåæç对象çç±»å彿°
- ä¼ éç¬å çæ¥æå¨æçåæçå¯¹è±¡çæææå°å½æ°
- ä»å½æ°è·å¾ç¬å çæ¥æå¨æçåæå¯¹è±¡çæææ
- ä½ä¸ºå ·ç§»å¨å®¹å¨çå ç´ ç±»åï¼ä¾å¦ä¿ææå卿åé 对象çæéç std::vector ï¼ä¾å¦ï¼è¥æ³è¦å¤æè¡ä¸ºï¼
std::unique_ptr å¯ä¸ºä¸å®æ´ç±»å T æé ï¼ä¾å¦ç¨äºæ¹åç¨ä½ pImpl ææ³ä¸æçç¨éãè¥ä½¿ç¨é»è®¤å é¤å¨ï¼å T å¿
é¡»å¨ä»£ç ä¸è°ç¨å é¤å¨ç¹å¤å®æ´ï¼è¿åçäºææå½æ°ãç§»å¨èµå¼è¿ç®ç¬¦å std::unique_ptr ç reset æå彿°ä¸ãï¼ç¸åå°ï¼ std::shared_ptr ä¸è½ä»æåä¸å®æ´ç±»åç裸æéæé ï¼ä½å¯äº T ä¸å®æ´å¤éæ¯ï¼ã注æè¥ T æ¯ç±»æ¨¡æ¿ç¹åï¼å以 unique_ptr 为è¿ç®æ°ç使ç¨ï¼å¦ !p ï¼å ADL èè¦æ± T çå½¢å宿´ã
è¥ T æ¯æåºç±» B ç导åºç±»ï¼å std::unique_ptr<T> å¯éå¼è½¬æ¢ä¸º std::unique_ptr<B>ã产çç std::unique_ptr<B> çé»è®¤å é¤å¨å°ä½¿ç¨ B ç operator delete ï¼è¿å¯¼è´æªå®ä¹è¡ä¸ºï¼é¤é B çææå½æ°ä¸ºèãæ³¨æ std::shared_ptr è¡¨ç°æå«ï¼ std::shared_ptr<B> å°ä½¿ç¨ç±»å T ç operator delete ï¼èä¸å³ä½¿ B çææå½æ°éèï¼ä¹ä¼æ£ç¡®å é¤è¢«å æå¯¹è±¡ã
ä¸åäº std::shared_ptr ï¼ std::unique_ptr å¯éè¿ä»»ä½æ»¡è¶³å¯ç©ºæé (NullablePointer) çå®å¶æç±»å管ç对象ãä¾å¦ï¼è¿å
许管çä½äºå
±äº«å
åï¼ä½æä¾å®ä¹ typedef boost::offset_ptr pointer; æå
¶ä»ç¼é¥°æéç Deleter ç对象ã
æåç±»å
| Â | |
| æåç±»å | å®ä¹ |
| pointer | è¥è¯¥ç±»ååå¨å为 std::remove_reference<Deleter>::type::pointer ï¼å¦å为 T* ãå¿
须满足å¯ç©ºæé (NullablePointer) ã
|
| element_type | T ï¼æ¤ unique_ptr æç®¡çç对象类å
|
| deleter_type | Deleter ï¼å½æ°å¯¹è±¡æå°å½æ°æå°å½æ°å¯¹è±¡çå·¦å¼å¼ç¨ï¼ä¼ä»ææå½æ°è°ç¨
|
æå彿°
æé æ°çunique_ptr (å ¬å¼æå彿°) | |
| æææç®¡çç对象ï¼å¦æåå¨çè¯ (å ¬å¼æå彿°) | |
为unique_ptrèµå¼ (å ¬å¼æå彿°) | |
ä¿®æ¹å¨ | |
| è¿åä¸ä¸ªæå被管ç对象çæéï¼å¹¶éæ¾æææ (å ¬å¼æå彿°) | |
| æ¿æ¢è¢«ç®¡ç对象 (å ¬å¼æå彿°) | |
| 交æ¢è¢«ç®¡ç对象 (å ¬å¼æå彿°) | |
è§å¯å¨ | |
| è¿åæå被管ç对象çæé (å ¬å¼æå彿°) | |
| è¿åç¨äºææè¢«ç®¡ç对象çå é¤å¨ (å ¬å¼æå彿°) | |
| æ£æ¥æ¯å¦æå
³èç被管ç对象 (å ¬å¼æå彿°) | |
åå¯¹è±¡çæ¬ï¼
| |
| è§£å¼ç¨æå被管ç对象çæé (å ¬å¼æå彿°) | |
æ°ç»çæ¬ï¼
| |
| æä¾å°è¢«ç®¡çæ°ç»çæç´¢å¼è®¿é® (å ¬å¼æå彿°) | |
éæå彿°
| (C++14)(C++20) |
å建管çä¸ä¸ªæ°å¯¹è±¡çç¬å æé (彿°æ¨¡æ¿) |
| (C++20 ä¸ç§»é¤)(C++20) |
ä¸å¦ä¸ä¸ª unique_ptr æ nullptr è¿è¡æ¯è¾ (彿°æ¨¡æ¿) |
| (C++20) |
è¾åºè¢«ç®¡çæéçå¼å°è¾åºæµ (彿°æ¨¡æ¿) |
| (C++11) |
ç¹å std::swap ç®æ³ (彿°æ¨¡æ¿) |
è¾ å©ç±»
| (C++11) |
std::unique_ptr çæ£åæ¯æ (类模æ¿ç¹å) |
示ä¾
#include <iostream> #include <vector> #include <memory> #include <cstdio> #include <fstream> #include <cassert> #include <functional>  struct B { virtual void bar() { std::cout << "B::bar\n"; } virtual ~B() = default; }; struct D : B { D() { std::cout << "D::D\n"; } ~D() { std::cout << "D::~D\n"; } void bar() override { std::cout << "D::bar\n"; } };  // æ¶è´¹ unique_ptr ç彿°è½ä»¥å¼æä»¥å³å¼å¼ç¨æ¥æ¶å® std::unique_ptr<D> pass_through(std::unique_ptr<D> p) { p->bar(); return p; }  void close_file(std::FILE* fp) { std::fclose(fp); }  int main() { std::cout << "unique ownership semantics demo\n"; { auto p = std::make_unique<D>(); // p æ¯å æ D ç unique_ptr auto q = pass_through(std::move(p)); assert(!p); // ç°å¨ p ä¸å æä»»ä½å å®¹å¹¶ä¿æç©ºæé q->bar(); // è q å æ D 对象 } // ~D è°ç¨äºæ¤  std::cout << "Runtime polymorphism demo\n"; { std::unique_ptr<B> p = std::make_unique<D>(); // p æ¯å æ D ç unique_ptr // ä½ä¸ºæååºç±»çæé p->bar(); // èæ´¾å  std::vector<std::unique_ptr<B>> v; // unique_ptr è½åå¨äºå®¹å¨ v.push_back(std::make_unique<D>()); v.push_back(std::move(p)); v.emplace_back(new D); for(auto& p: v) p->bar(); // èæ´¾å } // ~D called 3 times  std::cout << "Custom deleter demo\n"; std::ofstream("demo.txt") << 'x'; // åå¤è¦è¯»çæä»¶ { std::unique_ptr<std::FILE, void (*)(std::FILE*) > fp(std::fopen("demo.txt", "r"), close_file); if(fp) // fopen å¯ä»¥æå¼å¤±è´¥ï¼è¯¥æ åµä¸ fp ä¿æç©ºæé std::cout << (char)std::fgetc(fp.get()) << '\n'; } // fclose() è°ç¨äºæ¤ï¼ä½ä» è¥ FILE* 䏿¯ç©ºæé // ï¼å³ fopen æåï¼ Â std::cout << "Custom lambda-expression deleter demo\n"; { std::unique_ptr<D, std::function<void(D*)>> p(new D, [](D* ptr) { std::cout << "destroying from a custom deleter...\n"; delete ptr; }); // p å æ D p->bar(); } // è°ç¨ä¸è¿° lambda 并鿝 D  std::cout << "Array form of unique_ptr demo\n"; { std::unique_ptr<D[]> p{new D[3]}; } // è°ç¨ ~D 3 次 }
è¾åºï¼
unique ownership semantics demo D::D D::bar D::bar D::~D Runtime polymorphism demo D::D D::bar D::D D::D D::bar D::bar D::bar D::~D D::~D D::~D Custom deleter demo x Custom lambda-expression deleter demo D::D D::bar destroying from a custom deleter... D::~D Array form of unique_ptr demo D::D D::D D::D D::~D D::~D D::~D