unique_ptr 是 c++11 引入的独占式智能 指针 ,通过 move 语义转移所有权,配合 make_unique 可安全高效管理动态 对象,防止内存泄漏。

在 C ++ 中,unique_ptr 是一种智能指针,用于管理动态分配的对象,确保同一时间只有一个指针拥有该对象的所有权。当 unique_ptr 被销毁时,它所指向的对象也会自动被释放,从而有效防止内存泄漏。它是 C++11 引入的资源管理 工具 之一,适用于需要独占所有权的场景。
基本定义与初始化
要使用 unique_ptr,需要包含头文件 <memory>。它的基本语法如下:
std::unique_ptr<T> ptr; std::unique_ptr<T> ptr(new T); auto ptr = std::make_unique<T>(……); // 推荐方式(C++14 起)
建议优先使用 std::make_unique,因为它更安全、更高效,并能避免重复调用 new。
示例:
立即学习“C++ 免费学习笔记(深入)”;
#include <memory> #include <iostream> <p>struct MyClass {int value; MyClass(int v) : value(v) {std::cout << " 构造 n ";} ~MyClass() { std::cout << " 析构 n ";} };</p><p>auto ptr = std::make_unique<MyClass>(42); // 自动管理生命周期 </p>
所有权唯一,不可复制
unique_ptr 的核心特性是独占所有权,因此不支持拷贝构造和拷贝赋值。
auto ptr1 = std::make_unique<int>(10); // auto ptr2 = ptr1; // 错误:不能复制 auto ptr2 = std::move(ptr1); // 正确:通过 move 转移所有权
执行 move 操作后,ptr1 变为 nullptr,ptr2 拥有对象。这是实现资源安全转移的关键机制。
作为函数参数和返回值
在函数间传递 unique_ptr 时,通常使用移动语义。
作为返回值:
std::unique_ptr<MyClass> createObject() { return std::make_unique<MyClass>(100); } <p>auto obj = createObject(); // 接收所有权 </p>
作为参数(若需转移所有权):
void takeOwnership(std::unique_ptr<MyClass> ptr) {std::cout << ptr->value << "n";} // 函数结束时自动析构 <p>takeOwnership(std::move(obj)); // obj 失去所有权 </p>
如果只是临时访问对象,应使用指针或引用:
void useobject(const MyClass* ptr) {if (ptr) std::cout << ptr->value << "n"; } // 或 void useObject(const MyClass& ref) {std::cout << ref.value << "n";}
自定义删除器
默认情况下,unique_ptr 使用 delete 释放资源。对于特殊资源(如 FILE*、C API 对象),可指定自定义删除器。
// 管理 C 风格文件 auto file_deleter = [](FILE* f) {if (f) fclose(f); }; std::unique_ptr<FILE, decltype(file_deleter)> file(fopen("test.txt", "w"), file_deleter); <p>if (file) {fprintf(file.get(), "Hellon"); } // 离开 作用域 时自动关闭文件 </p>
自定义删除器可以是函数指针、Lambda 或仿函数,使 unique_ptr 更具扩展性。
基本上就这些。unique_ptr 设计简洁,强调“单一所有权”,配合 move 语义和 make_unique,能写出既安全又高效的代码。只要记住它不能复制,需要用 move 来转移,日常使用就不会出错。


