c++中如何使用unique_ptr_C++智能指针unique_ptr用法详解

std::unique_ptr是C++中用于独占式管理动态对象的智能指针,定义于头文件。1. 推荐使用std::make_unique创建,安全且高效;2. 独占所有权,禁止拷贝,只能通过std::move转移;3. 支持自定义删除器,适用于数组、文件等特殊资源;4. 可作为函数返回值或参数传递所有权,接收方用move获取;5. get()获取裸指针但不释放,release()则释放所有权需手动delete;6. 常用于RAII、工厂模式和异常安全场景,确保资源自动释放,避免内存泄漏。

c++中如何使用unique_ptr_c++智能指针unique_ptr用法详解

在C++中,std::unique_ptr 是一种智能指针,用于管理动态分配的对象,确保该对象在其作用域结束时自动被释放。它实现了独占式的所有权语义——即同一时间只有一个 unique_ptr 可以拥有某个对象,防止资源的重复释放或内存泄漏。

1. 基本用法与创建方式

unique_ptr 定义在头文件 中,使用前需包含此头文件。

最简单的创建方式是使用 std::make_unique(C++14 起支持):

#include #include 

int main() {auto ptr = std::make_unique(42);std::cout << *ptr << "n"; // 输出: 42return 0;}

如果不能使用 C++14,也可以直接构造:

立即学习“C++免费学习笔记(深入)”;

std::unique_ptr ptr(new int(42));

但推荐使用 make_unique,因为它更安全、更高效,并能避免一些异常安全问题。

2. 独占所有权特性

unique_ptr 不允许拷贝构造和拷贝赋值,因为其所有权是独占的。

auto ptr1 = std::make_unique(10);// auto ptr2 = ptr1;        // 错误:不能拷贝// auto ptr3 = ptr1.get();  // 错误:裸指针不接管所有权

auto ptr4 = std::move(ptr1); // 正确:通过 move 转移所有权// 此时 ptr1 为空,ptr4 拥有对象

调用 std::move 后,原指针变为 null,不能再访问其所指向的对象。

3. 自定义删除器

默认情况下,unique_ptr 使用 delete 释放资源,但你可以指定自定义删除器来处理特殊场景,比如释放数组、调用特定函数等。

// 处理数组auto array_ptr = std::make_unique(10); // C++14 起支持for (int i = 0; i < 10; ++i) array_ptr[i] = i;

// 或者手动指定删除器(如关闭文件)auto file_deleter = [](FILE* f) {if (f) fclose(f);};std::unique_ptr fp(fopen("test.txt", "r"), file_deleter);

// 使用 lambda 作为删除器时需显式指定类型

4. 作为函数参数和返回值

unique_ptr 非常适合用于函数间传递资源,尤其是工厂模式中返回动态对象。

std::unique_ptr create_value() {    return std::make_unique(100);}

void use_value(std::unique_ptr data) {std::cout << *data << "n";}

int main() {auto p = create_value(); // 接收所有权use_value(std::move(p)); // 传递所有权// 此时 p 已为空return 0;}

若函数只需读取数据而不获取所有权,应传 const 引用:

void inspect(const std::unique_ptr& ptr) {    if (ptr) std::cout << *ptr << "n";}

5. 与裸指针交互

可通过 get() 获取底层裸指针,但不会转移所有权:

auto ptr = std::make_unique(42);int* raw = ptr.get();  // raw 指向同一地址*raw = 100;std::cout << *ptr;     // 输出: 100

也可用 release() 主动释放所有权并返回裸指针:

int* released_ptr = ptr.release(); // ptr 变为 nullptr,不释放内存// 必须手动 delete released_ptr 否则会内存泄漏delete released_ptr;

6. 常见应用场景

替代原始指针实现 RAII 资源管理类中管理动态成员变量,避免手动析构工厂函数返回动态对象异常安全代码中防止资源泄漏

基本上就这些。合理使用 unique_ptr 能显著提升代码的安全性和可维护性,减少内存错误。只要记住:谁拥有,谁释放;转移用 move;别拷贝。不复杂但容易忽略细节。

以上就是c++++中如何使用unique_ptr_C++智能指针unique_ptr用法详解的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1476440.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 00:24:54
下一篇 2025年12月19日 00:25:06

相关推荐

  • C++如何优化STL算法调用效率

    选择合适的STL容器需根据算法特性与操作需求权衡,如std::vector适合随机访问和排序,而频繁插入删除则优选std::list或std::deque以提升效率。 优化C++ STL算法调用效率的关键在于深入理解算法的时间复杂度、空间复杂度,以及它们如何与底层容器的迭代器特性协同工作。这通常意味…

    2025年12月19日
    000
  • c++怎么定义一个函数指针_C++函数指针的定义与使用方法

    函数指针用于存储函数地址并实现动态调用。定义格式为返回类型(指针名)(参数列表),如int (funcPtr)(int, int);可指向匹配签名的函数,如add或subtract;通过funcPtr(3, 4)或(*funcPtr)(3, 4)调用,前者更常用;常用于回调机制和算法选择,如calc…

    2025年12月19日
    000
  • c++中预处理器指令有哪些_常用预处理器指令全面总结

    C++预处理器指令以#开头,用于宏定义、文件包含、条件编译等。1. #include包含头文件,从标准路径查找,””优先当前目录;2. #define定义常量或函数宏,如#define PI 3.14;3. #undef取消宏定义,防止宏污染;4. 条件编译指令#ifdef、…

    2025年12月19日
    000
  • c++中#include “” 和 #include 的区别_头文件包含路径搜索机制辨析

    include “” 优先在当前源文件目录查找,适用于项目内部头文件;2. #include 仅在系统标准路径查找,用于标准库或第三方库;3. 正确区分使用可避免包含错误并提升构建稳定性。 的区别_头文件包含路径搜索机制辨析”> 在C++中,#include…

    2025年12月19日
    000
  • c++中如何使用dynamic_cast_dynamic_cast动态类型转换指南

    dynamic_cast用于多态类型的安全转换,通过运行时检查确保类型安全。1. 语法为dynamic_cast(源指针)或dynamic_cast(源引用),目标类需含虚函数。2. 指针转换失败返回nullptr,引用失败抛出std::bad_cast异常。3. 使用条件包括:类必须为多态类型、仅…

    2025年12月19日
    000
  • c++中memset函数的使用方法_c++ memset函数用法与注意事项

    答案:memset是C++中按字节初始化内存的函数,适用于POD类型,常用于将数组置0或-1,但不可用于非0/−1的整型赋值及复杂对象,易因字节填充误解导致错误,现代C++推荐使用std::fill或容器初始化等更安全的方式替代。 在C++中,memset 是一个用于内存块初始化的函数,常用于将一段…

    2025年12月19日
    000
  • c++中头文件怎么写_C++头文件编写规范

    答案:C++头文件应使用头文件保护(如#pragma once或宏定义)防止重复包含,仅包含声明而非定义,合理使用前置声明减少依赖,按规则顺序包含必要头文件,避免using指令污染全局命名空间,模板和内联函数可定义在头文件中,并添加清晰注释以提升可维护性。 在C++中,头文件(.h 或 .hpp)用…

    2025年12月19日 好文分享
    000
  • c++中lambda表达式怎么用_c++ lambda表达式使用方法详解

    lambda表达式用于定义匿名函数,语法为[捕获列表](参数列表)->返回类型{函数体},可简洁传递给STL算法等。 在C++中,lambda表达式是一种定义匿名函数的简洁方式,常用于需要传递函数作为参数的场景,比如STL算法中的sort、find_if等。它让代码更直观、紧凑,尤其适合写短小…

    2025年12月19日
    000
  • c++中数组怎么传递给函数_C++函数数组参数传递

    一维数组作为函数参数时,实际传递的是首地址,形参可写为指针或空下标数组形式,如void printArray(int arr[], int size)。 在C++中,数组传递给函数时不能直接按值传递整个数组(除非使用std::array或封装在结构体中),通常采用的是将数组的首地址传递给函数。这意味…

    2025年12月19日
    000
  • c++中怎么实现回调函数_C++回调函数设计与实现方法

    C++中实现回调函数的核心是将函数作为参数传递,常用方法包括函数指针、std::function、lambda表达式和类成员函数绑定。函数指针适用于普通函数或静态成员函数,通过定义函数指针类型并传参实现回调;std::function结合lambda可支持闭包和多种可调用对象,灵活性更高;类成员函数…

    2025年12月19日
    000
  • c++中宏定义#define怎么用_c++ #define宏定义使用指南

    define用于定义宏,可定义常量如PI、MAX_SIZE,但无类型检查,推荐用const或constexpr替代。2. 可定义带参宏如SQUARE(x),需注意括号防止展开错误。3. 与#ifdef、#ifndef配合实现条件编译,防止头文件重复包含。4. 用#undef可删除宏定义以限制作用域。…

    2025年12月19日
    000
  • c++中深拷贝和浅拷贝有什么区别_深拷贝与浅拷贝的对比分析

    深拷贝会为指针成员分配新内存并复制数据,确保对象独立;浅拷贝仅复制指针地址,导致多个对象共享同一内存,可能引发释放异常或数据错误。 在C++中,深拷贝和浅拷贝的区别主要体现在对象复制时对指针所指向内存的处理方式。当类中含有动态分配的成员变量(如指针)时,这个区别尤为关键。 什么是浅拷贝 浅拷贝是指只…

    2025年12月19日
    000
  • c++中什么是完美转发_std::forward与完美转发实现机制

    std::forward是实现完美转发的标准库函数,通过模板参数推导和引用折叠规则,将通用引用T&&保持原始值类别传递给目标函数。 在C++中,完美转发是指在模板函数中将参数以完全相同的值类别(左值或右值)传递给另一个函数的能力。也就是说,如果传入的是左值,转发时就保持为左值;如果传…

    2025年12月19日
    000
  • 如何在C++中链接一个外部库_C++外部库链接配置方法

    整合C++外部库需配置头文件和库文件路径,通过命令行、IDE或CMake实现;推荐使用CMake,因其跨平台、支持自动查找库(find_package)和管理依赖,简化构建过程。 在C++项目中整合外部库,本质上就是告诉编译器和链接器去哪里找你声明和定义的那些函数、类或变量。这通常涉及两个核心步骤:…

    2025年12月19日
    000
  • C++内存模型与并发容器实现原理

    C++内存模型通过原子操作和内存顺序保证多线程数据一致性,并发容器则基于此实现线程安全;原子操作如atomic_int确保操作不可分割,避免竞态条件;常见并发容器有基于锁、无锁和分段锁三种,分别在安全性与性能间权衡;避免死锁需按序加锁或使用std::scoped_lock;合理选择memory_or…

    2025年12月19日
    000
  • c++怎么获取cpu核心数_cpu核心数获取方法

    答案是使用std::thread::hardware_concurrency()获取CPU核心数,该方法跨平台且返回逻辑核心数;若返回0表示信息不可用,建议结合Windows的GetSystemInfo或Linux的sysconf进行平台特定处理。 在C++中获取CPU核心数,最常用且跨平台的方法是…

    2025年12月19日
    000
  • c++中怎么删除map中的一个元素_map中erase方法使用详解

    std::map::erase 提供三种删除方式:通过迭代器、键值或范围删除;使用时需注意迭代器失效、避免对 end() 操作,并合理利用返回值判断删除结果,确保安全高效操作。 在C++中,std::map 是一个关联容器,用于存储键值对,并按照键的顺序自动排序。当我们需要从 map 中删除元素时,…

    2025年12月19日
    000
  • c++中virtual关键字的作用_virtual关键字实现运行时多态

    virtual关键字用于声明虚函数以实现运行时多态;2. 通过基类指针或引用调用虚函数时,根据实际对象类型动态调用对应版本;3. 派生类可重写基类的虚函数以实现不同行为。 在C++中,virtual关键字用于实现运行时多态,这是面向对象编程的重要特性之一。它的主要作用是允许派生类重写(overrid…

    2025年12月19日
    000
  • c++如何连接两个字符串_c++字符串拼接操作全攻略

    C++中连接字符串最直接的方式是使用+或+=运算符,其中+会创建新对象,而+=在原字符串上修改更高效。此外,append()方法功能更丰富,支持部分拼接和重复字符追加;std::stringstream适合拼接多种数据类型,类型安全且灵活;C++20引入的std::format兼具高性能与高可读性,…

    2025年12月19日
    000
  • c++中dynamic_cast和static_cast的区别_C++ dynamic_cast与static_cast转换区别详解

    dynamic_cast在运行时进行安全的向下转型,依赖RTTI检查类型,转换失败返回nullptr或抛异常,要求类有多态性;static_cast在编译期完成转换,无运行时开销,适用于已知安全的场景如向上转型或基本类型转换,但向下转型时不检查类型,错误使用导致未定义行为。两者均需继承关系,不可用于…

    2025年12月19日
    000

发表回复

登录后才能评论
关注微信