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

小万哥
• 阅读 498

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

什么是智能指针?

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

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

C++中的几种智能指针

C++中有三种智能指针:unique_ptrshared_ptrweak_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;
 }

data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

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;
 }

data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

weak_ptr

weak_ptr是一个弱引用的智能指针,它可以与shared_ptr一起使用。weak_ptr不会增加所管理的对象的引用计数,因此它不会影响对象的生命周期。可以通过weak_ptrlock()成员函数来获取一个指向所管理的对象的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;
 }

data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

总结

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

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

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

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

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Wesley13 Wesley13
3年前
PPDB:今晚老齐直播
【今晚老齐直播】今晚(本周三晚)20:0021:00小白开始“用”飞桨(https://www.oschina.net/action/visit/ad?id1185)由PPDE(飞桨(https://www.oschina.net/action/visit/ad?id1185)开发者专家计划)成员老齐,为深度学习小白指点迷津。
Wesley13 Wesley13
3年前
VBox 启动虚拟机失败
在Vbox(5.0.8版本)启动Ubuntu的虚拟机时,遇到错误信息:NtCreateFile(\\Device\\VBoxDrvStub)failed:0xc000000034STATUS\_OBJECT\_NAME\_NOT\_FOUND(0retries) (rc101)Makesurethekern
Wesley13 Wesley13
3年前
FLV文件格式
1.        FLV文件对齐方式FLV文件以大端对齐方式存放多字节整型。如存放数字无符号16位的数字300(0x012C),那么在FLV文件中存放的顺序是:|0x01|0x2C|。如果是无符号32位数字300(0x0000012C),那么在FLV文件中的存放顺序是:|0x00|0x00|0x00|0x01|0x2C。2.  
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
PHP创建多级树型结构
<!lang:php<?php$areaarray(array('id'1,'pid'0,'name''中国'),array('id'5,'pid'0,'name''美国'),array('id'2,'pid'1,'name''吉林'),array('id'4,'pid'2,'n
Wesley13 Wesley13
3年前
Java日期时间API系列36
  十二时辰,古代劳动人民把一昼夜划分成十二个时段,每一个时段叫一个时辰。二十四小时和十二时辰对照表:时辰时间24时制子时深夜11:00凌晨01:0023:0001:00丑时上午01:00上午03:0001:0003:00寅时上午03:00上午0
Wesley13 Wesley13
3年前
MBR笔记
<bochs:100000000000e\WGUI\Simclientsize(0,0)!stretchedsize(640,480)!<bochs:2b0x7c00<bochs:3c00000003740i\BIOS\$Revision:1.166$$Date:2006/08/1117
Stella981 Stella981
3年前
Jenkins 插件开发之旅:两天内从 idea 到发布(上篇)
本文首发于:Jenkins中文社区(https://www.oschina.net/action/GoToLink?urlhttp%3A%2F%2Fjenkinszh.cn)!huashan(https://oscimg.oschina.net/oscnet/f499d5b4f76f20cf0bce2a00af236d10265.jpg)
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_