C++的智能指针是RAII的一种应用 自动管理动态内存的工具 可以避免显示的内存释放 减少内存泄漏和为定义行为的可能性 C++标准库提供了三种主要的智能指针 unique_ptr shared_ptr weak_ptr
可能有的同学听过auto_ptr 这个东西在C++11被弃用 在C++17被移除 主要是由于复制问题 可能会导致内存重复释放
unique_ptr
这个智能指针对所指对象是独占所有权的 也就是说 同一时刻 只有一个unique_ptr指向这个对象
他不允许复制 但是可以使用move变为右值 再移动赋值 将这个对象的所有权转移到另一个unique_ptr
这个智能指针只需要管理一个对象 开销也比较小 适用于高性能的场景 比如说文件具柄 数据库连接等
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| #include <memory>
std::unique_ptr<int> createUniquePtr() { return std::make_unique<int>(10); }
int main() { std::unique_ptr<int> ptr = createUniquePtr(); std::unique_ptr<int> newPtr = std::move(ptr); }#include <memory>
std::unique_ptr<int> createUniquePtr() { return std::make_unique<int>(10); }
int main() { std::unique_ptr<int> ptr = createUniquePtr(); std::unique_ptr<int> newPtr = std::move(ptr); }
|
shared_ptr
这个智能指针是共享对象的所有权的 是使用引用计数来维护同一个对象 当最后一个shared_ptr被销毁时 对象才会被释放
引用计数是线程安全的 可以在多线程下使用 比较适合在共享所有权的情况下使用 管理一些生命周期复杂 不确定何时释放的情况比较好用
1 2 3 4 5 6 7 8 9 10 11 12
| #include <memory> #include <iostream>
void useShared(std::shared_ptr<int> sptr) { std::cout << "Use count: " << sptr.use_count() << std::endl; }
int main() { std::shared_ptr<int> ptr1 = std::make_shared<int>(10); std::shared_ptr<int> ptr2 = ptr1; useShared(ptr1); }
|
weak_ptr
这是一种弱引用 主要是和shared_ptr配合使用的 他不影响对象的引用计数 也就说他并不会独立拥有对象本身
他的作用主要是打破循环引用的场景 防止多个对象通过shared_ptr互相引用 从而导致内存无法释放 主要用于缓存或者观察者模式中 允许对象访问但不拥有其生命周期
1 2 3 4 5 6 7 8 9 10 11 12 13
| #include <memory> #include <iostream>
int main() { std::shared_ptr<int> sptr = std::make_shared<int>(10); std::weak_ptr<int> wptr = sptr;
if (auto spt = wptr.lock()) { std::cout << *spt << std::endl; } else { std::cout << "Object no longer exists" << std::endl; } }
|