C++ 智能指针和内存管理:使用指南和技巧

2023-12-05

C++是一门强大的编程语言,但是在内存管理方面却存在着一些问题。手动管理内存不仅费时费力,而且容易出错。因此,C++中引入了智能指针这一概念,以更好地管理内存。

什么是智能指针?

在C++中,内存的分配和释放都是由开发者手动实现的。这种方式虽然很灵活,但也十分容易出错,比如忘记释放内存或释放了已经释放的内存等。为了避免这些问题,C++引入了智能指针这一概念。智能指针是一种类,它在析构时自动释放所管理的对象所占用的内存。这样,程序员就不需要手动管理内存,减少了出错的可能性。智能指针是一种 RAII (Resource Acquisition Is Initialization)技术的应用。

RAII 的基本思想是:在对象的构造函数中进行资源的分配,在析构函数中进行资源的释放。智能指针也是这种思想的一种扩展,它在析构时自动释放资源。

C++中的几种智能指针

C++中有三种智能指针: unique_ptr shared_ptr weak_ptr 。每种智能指针都有其独特的功能和特点,下面将逐一介绍。

unique_ptr

unique_ptr 是一个独享所有权的智能指针,不能共享所有权。当 unique_ptr 被销毁时,它所管理的对象的内存也会被自动释放。 unique_ptr 也可以通过 std::move() 转移所有权。 unique_ptr 使用的方法很简单,只需要将所需管理的对象传递给 unique_ptr 即可。

  #include <iostream>
  #include <memory>
 
  int main() {
      // 使用unique_ptr管理int类型的对象
      std::unique_ptr<int> up1(new int(10));
      std::cout << "up1: " << *up1 << std::endl;
 
      // 使用make_unique函数创建unique_ptr对象
      auto up2 = std::make_unique<int>(20);
      std::cout << "up2: " << *up2 << std::endl;
 
      // unique_ptr可以通过std::move()转移所有权
      std::unique_ptr<int> up3 = std::move(up1);
      std::cout << "up3: " << *up3 << std::endl;
 
      return 0;
  }

shared_ptr

shared_ptr 是一个共享所有权的智能指针,可以有多个 shared_ptr 指向同一个对象。每当一个 shared_ptr 被销毁时,它所管理的对象的引用计数会减1。当引用计数为0时,对象的内存也会被自动释放。 shared_ptr 的使用方法和 unique_ptr 类似,只需要将所需管理的对象传递给 shared_ptr 即可。需要注意的是, shared_ptr 不能管理动态分配的数组,因为它无法确定数组的长度。

  #include <iostream>
  #include <memory>
 
  int main() {
      // 使用shared_ptr管理int类型的对象
      std::shared_ptr<int> sp1(new int(10));
      std::cout << "sp1: " << *sp1 << std::endl;
 
      // 使用make_shared函数创建shared_ptr对象
      auto sp2 = std::make_shared<int>(20);
      std::cout << "sp2: " << *sp2 << std::endl;
 
      // 可以有多个shared_ptr指向同一个对象
      std::shared_ptr<int> sp3 = sp1;
      std::cout << "sp1 count: " << sp1.use_count() << std::endl;
      std::cout << "sp3 count: " << sp3.use_count() << std::endl;
 
      return 0;
  }

weak_ptr

weak_ptr 是一个弱引用的智能指针,它可以与 shared_ptr 一起使用。 weak_ptr 不会增加所管理的对象的引用计数,因此它不会影响对象的生命周期。可以通过 weak_ptr lock() 成员函数来获取一个指向所管理的对象的 shared_ptr 。需要注意的是,在使用 lock() 函数之前,需要判断 weak_ptr 是否已经过期,即判断其指向的对象是否已经被销毁。

  #include <iostream>
  #include <memory>
 
  int main() {
      // 使用shared_ptr管理int类型的对象
      std::shared_ptr<int> sp1(new int(10));
      std::weak_ptr<int> wp1 = sp1;
 
      // 判断wp1是否过期
      if (auto sp2 = wp1.lock()) {
          std::cout << "wp1: " << *sp2 << std::endl;
      } else {
          std::cout << "wp1 expired" << std::endl;
      }
 
      // 销毁sp1
      sp1.reset();
 
      // 判断wp1是否过期
      if (auto sp2 = wp1.lock()) {
          std::cout << "wp1: " << *sp2 << std::endl;
      } else {
          std::cout << "wp1 expired" << std::endl;
      }
 
      return 0;
  }

总结

智能指针是C++中一种非常实用的内存管理工具。它可以帮助程序员自动管理内存,减少出错的可能性。C++中有三种智能指针: unique_ptr shared_ptr weak_ptr 。每种智能指针都有其特点,程序员可以根据实际情况选择使用。

在使用智能指针时,需要注意以下几点:

  • 不要将普通指针和智能指针混用,避免重复释放内存或内存泄漏。
  • 不要将同一个对象交给不同的智能指针管理,避免引用计数出现错误。
  • shared_ptr 不能管理动态分配的数组,因为它无法确定数组的长度。
  • 在使用 weak_ptr lock() 函数之前,需要判断 weak_ptr 是否已经过期,即判断其指向的对象是否已经被销毁。

使用智能指针可以大大提高代码的可读性和可维护性,建议大家在编写C++程序时多加使用。

最后

为了方便其他设备和平台的小伙伴观看往期文章:

微信公众号搜索: Let us Coding ,关注后即可获取最新文章推送

看完如果觉得有帮助,欢迎 点赞、收藏、关注

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++ 智能指针和内存管理:使用指南和技巧 的相关文章

  • 如何修复 scanf 将数据放入数组中?

    有人可以就 scanf 提出建议吗 message 是一个 4 行 x 16 列的数组 用户输入数字 全部为 1 位整数 后 按 Enter 键时会出现错误消息 正如所说 可能与 scanf 有关 for int i 0 i lt M i
  • pthread_create 编译返回错误

    我使用以下代码创建两个线程 header files include
  • 为什么 strcat() 之后字符串会被改变?

    这是源代码 int main char str dance char str1 hello char str2 abcd strcat str1 str2 printf s str output bcd why str更改后strcat s
  • 将 void* 作为函数调用而不声明函数指针

    我已经搜索过 但找不到任何结果 我的术语可能有问题 所以如果以前有人问过这个问题 请原谅我 我想知道是否有一种简单的方法可以调用void 作为 C 中的函数 无需首先声明函数指针 然后为函数指针分配地址 IE 假设要调用的函数是类型void
  • 调试Windows服务

    Scenario 我有一个用 C 编写的 Windows 服务 我已经阅读了所有关于如何调试它的谷歌线程 但我仍然无法让它工作 我已经运行 PathTo NetFramework InstallUtil exe C MyService ex
  • Xamarin 中的 Task.ConfigureAwait(false) - 安全使用/建议使用?

    经验法则是 如果它不是与 UI 相关的方法 请使用Task ConfigureAwait false 如果我有一个接受接口的 PCL 核心库怎么办IUIAccess 核心库中的视图模型有一个方法 public Task ViewModelL
  • 是否可以将向量的一部分作为向量发送给函数? [复制]

    这个问题在这里已经有答案了 我想看看是否可以将向量的一部分传递给函数 以便它显示为函数的法线向量 更重要的是 我希望这可以在 O 1 的常数时间内完成 我不想迭代向量来创建一个新向量 事实上 我还希望在下面的示例中将新向量的大小更改为 40
  • 如何修改道路网络的 L 系统?

    向大家问好 我目前正在研究道路网络的程序生成 并偶然发现了 L 系统算法 根据我从有关该主题的各种科学论文以及有关该主题的论文的进一步论文中了解到 算法更改为使用 全局目标和局部约束 其中修改所采取的路径以适应地形等输入值和人口密度 现在我
  • 如果文本框不为空,如何添加并显示工具提示文本框 WPF

    需要显示提示 其中包含文本字段中的数据 文本框有数据时出现提示 只需使用绑定到 ToolTipService 附加属性即可 XAML
  • 使用 JsonWriter 时,WriteStartConstructor 的用途是什么?

    标题说明了一切 我看到它 及其相应的结尾 吐出以下内容 new Foo 但我不明白什么new实际上是在反序列化时执行的 文档只是说它编写了一个 Json 构造函数 但没有说 Json 构造函数是什么is 此方法是作为增强功能的一部分引入的
  • 无法使用 process.ErrorDataReceived c# 获取进程错误输出

    我已经建立了Form我使用了一段时间的应用程序 现在我想捕捉StandardError我的流程及其standartOutput 我查看了答案SO and MSDN https msdn microsoft com en us library
  • MDI应用程序中父窗体的问题

    我使用按钮作为容器中的控件 父窗体 当子窗体出现时 父窗体中的控件 按钮 图片 标签 出现在子窗体上并将其覆盖 我看不到子窗体 有谁知道如何防止这种情况 我不想将这些控件设置为 Control Visible false 因为当我最小化子表
  • 如何以编程方式区分不同的 IOException?

    我正在对写入 Process 对象的 StandardInput 流的代码进行一些异常处理 Process 有点像 unix head 命令 它只读取输入流的一部分 当进程终止时 写入线程会失败并显示 IOException The pip
  • std::vector 错误 C2582:“operator =”函数在以下位置不可用

    我使用简单的向量push back到类型A的对象 并收到此错误 这是我的代码 class A public A int a int b int c include A h std vector a vec objects new std v
  • 不区分大小写“包含(字符串)”

    有没有办法让下面的返回为真 string title ASTRINGTOTEST title Contains string 似乎没有过载允许我设置区分大小写 目前我将它们都大写 但这很愚蠢 我指的是i18n http en wikiped
  • IBM Rhapsody 中状态图终止连接器的理解

    在IBM Rhapsody中 如果我使用new创建了一个类的实例 那么我们是否必须通过调用delete来处理内存的释放 或者Termination Connector将在其状态图中通过内存释放来处理其销毁 如果您使用 C 和 OXF 对象执
  • 如何从 .NET DataGridView 控件单元格值写入文本文件?

    我有以下代码应该循环遍历我的所有行DataGridView 并将其所有单元格值写入文本文件 但是 它输出所有行 但仅输出每行的第一个单元格 而不输出其他三个单元格 string file name C test1 txt var objWr
  • 如何在 C# 中获取包含表情符号的字符串的正确长度

    The 英语国旗表情符号 https emojipedia org flag for england 由 14 个字节的数据组成 组合后将呈现一个字符 如果我有如下所示的代码 var test ud83c udff4 udb40 udc67
  • OledbConnection.Dispose() 是否关闭连接? [复制]

    这个问题在这里已经有答案了 可能的重复 如果使用 using 子句 是否需要关闭 DbConnection https stackoverflow com questions 12033998 is there any need to cl
  • 缓存行对齐(需要文章澄清)

    我最近在我的应用程序中遇到了我认为是错误共享的问题 我查了一下关于如何将我的数据与缓存行对齐 他建议使用以下 C 代码 C using C 0x alignment syntax template

随机推荐