C++学习笔记(30):智能指针(unique_ptr)

核心本质

  • 独占所有权: 同一时间,只有一个unique_ptr 能管理同一个对象,禁止拷贝,只能移动
  • 自动释放: 离开作用域时自动调用delete / delete\[\],永不内存泄漏(正常流程下)
  • 零额外开销: 和原始指针性能几乎一致,没有引用计数等额外成本
    注:要包含 #include

unique_ptr的使用

我们可以使用unique_ptr 类的构造函数创建智能指针对象,来管理由 new,new\[\],创建出来的动态对象

unique_ptr 特性

unique_ptr特点:

  • 同时只能有一个unique_ptr 对象来持有动态对象
  • unique_ptr 对象不支持默认拷贝,默认赋值语义
  • unique_ptr 对象支持移动拷贝,移动赋值语义

因此,我们知道,unique_ptr 对象不能以值的方式作为函数参数,也不能存储到STL的容器中(容器要求元素必须能够拷贝)

arduino 复制代码
#if 0
#include <iostream>
#include <memory>
#include <vector>
using namespace std;

class Person
{
public:
    Person()
    {
        cout << "构造函数" << endl;
    }

    ~Person()
    {
        cout << "析构函数" << endl;
    }
};

void test()
{
    unique_ptr<Person> up1(new Person);
    unique_ptr<Person> up2(new Person);

    // 1. 禁止拷贝、赋值
    // unique_ptr(const unique_ptr&)            = delete;
    // unique_ptr& operator=(const unique_ptr&) = delete;

    // 2. 允许移动拷贝、赋值
    unique_ptr<Person> up3(move(up1));  // 移动拷贝
    up2 = move(up3);  // 移动赋值

    // 由此, unique_ptr 不允许作为容器元素
    // vector<unique_ptr<Person>> vec;
    // vec.push_back(up1);
}

int main()
{
    test();
    return 0;
}

#endif

unique_ptr 自定义删除器

unique_ptr 可用于管理 new 出来的动态对象,也可以管理其他需要手动关闭的资源。例如:文件对象。

由于 unique_ptr 默认使用 delete、delete\[\] 来释放被管理的资源。所以,当管理的对象不能通过 delete、delete\[\] 来释放时,就需要自定义删除器。

c 复制代码
    #if 1

#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
#include <functional>
using namespace std;

class Person
{
    public:
        Person()
    {
        cout << "构造函数" << endl;
    }

        ~Person()
    {
    cout << "析构函数" << endl;
    }
    };

struct Deleter
{
    void operator()(FILE* fp)
{
    cout << "文件被自动关闭" << endl;

    if (fp != nullptr)
    {
        fclose(fp);
        fp = nullptr;
        }
    }
};

void my_deleter(FILE* fp)
{
    cout << "文件被自动关闭" << endl;

    if (fp != nullptr)
{
    fclose(fp);
    fp = nullptr;
    }
}


void test()
{
// 1. 函数对象作为删除器
// unique_ptr<FILE, Deleter> up(fopen("./demo.txt", "w"), Deleter());
// unique_ptr<FILE, function<void(FILE*)>> up(fopen("./demo.txt", "w"),Deleter());

// 2. 普通函数作为删除器
// unique_ptr<FILE, decltype(&my_deleter)> up(fopen("./demo.txt", "w"), my_deleter);
// unique_ptr<FILE, void(*)(FILE *)> up(fopen("./demo.txt", "w"), my_deleter);
// unique_ptr<FILE, function<void(FILE*)>> up(fopen("./demo.txt", "w"), my_deleter);

// 3. 匿名函数作为删除器
unique_ptr<FILE, function<void(FILE *)>> up(fopen("./demo.txt", "w"), [](FILE *fp) {
        cout << "文件被自动关闭" << endl;

        if (fp != nullptr)
        {
            fclose(fp);
            fp = nullptr;
        }
    });


if (!up)
{
    cout << "文件打开失败" << endl;
    return;
}
fputs("hello world\n", up.get());
}

int main()
{
    test();
    return 0;
}

    #endif
相关推荐
h_a_o777oah8 分钟前
【算法专项】扩展域并查集:原理详解及解决大部分种类并查集问题(洛谷P5937 P2024 C++代码)
数据结构·c++·算法·acm·并查集·扩展域·逻辑建模
雾沉川1 小时前
Visual C++ 运行库合集 v105.0 部署与故障排查技术指南
开发语言·c++·dll
丘山望岳1 小时前
剑起霜华——平衡二叉树(AVL树 )精讲
开发语言·数据结构·c++
Boom_Shu2 小时前
浅拷贝与深拷贝
开发语言·c++·算法
Mortalbreeze2 小时前
C++ Lambda表达式详解:从捕获列表到底层原理
开发语言·c++
为何创造硅基生物2 小时前
LVGL
c++·ui
只做人间不老仙2 小时前
C++ grpc 拦截器示例学习
开发语言·c++·学习
qeen872 小时前
【C++】类与对象之零散知识点补充(四)
c++·笔记·学习·语法
Irissgwe3 小时前
顺序表和链表
数据结构·c++·链表·c·顺序表·线性表
牛油果子哥q3 小时前
二叉树(Binary Tree)零基础精讲,树基础概念、树形分类、核心性质、递归/层序遍历、完整代码与面试考点全解
c++·面试·数据挖掘