现代C++(Modern C++)基本用法实践:五、智能指针
概述
c++效率较高的一个原因是我们可以自己定制策略手动申请和释放内存,当然,也伴随着开发效率降低和内存泄漏的风险。为了减少手动管理内存带来的困扰,c++提出了智能指针,可以帮助我们进行内存管理,有三种:
std::unique_ptr是一种独占所有权的智能指针,它不允许多个指针指向同一个对象。std::unique_ptr是一种轻量级的智能指针,不需要额外的开销,因此在你不需要共享所有权的情况下,应优先使用std::unique_ptr。std::shared_ptr是一种可以共享所有权的智能指针。多个std::shared_ptr可以指向同一个对象,该对象只有在最后一个指向它的std::shared_ptr被销毁时才会被删除。std::shared_ptr使用引用计数来跟踪有多少个智能指针指向同一个对象。std::weak_ptr是一种弱引用智能指针,它可以指向std::shared_ptr所管理的对象,但它不会增加该对象的引用计数。这对于解决std::shared_ptr的循环引用问题很有用。
智能指针的类型决定了何时销毁和释放内存。当智能指针的生命周期结束时,它会自动清理其所有权的内存。这就避免了内存泄漏,使代码更安全、更健壮。
但是智能指针的使用,也要注意一些问题:
- 智能指针并不能自动解决循环引用的问题,需要手动在合适的场合使用
std::weak_ptr弱指针 - 智能指针的使用是“传染性”的,如果使用智能指针,要考虑好整个逻辑链条都使用智能指针而不是原生的指针
- 从实际项目使用上看,一些比较上层的模块使用智能指针,更加方便,而在底层模块更多的还是使用原生的指针,我想大概因为智能指针也是会带来一些开销而且可能和一些内存管理策略冲突(如某些优化内存策略可能不使用指针,而使用句柄来指向对象)
智能指针是一个对象,实现是非入侵式的。关于他们的大概的原理:
std::unique_ptr它包含一个原生指针,并在其析构函数中删除这个指针。std::unique_ptr还重载了->和*运算符,所以可以像使用原生指针一样使用std::unique_ptrstd::shared_ptr的实现则相对复杂一些。除了存储原生指针,std::shared_ptr还需要存储一个引用计数。每次创建一个新的std::shared_ptr或者调用std::shared_ptr的拷贝构造函数或赋值运算符时,引用计数就会增加。每次销毁一个std::shared_ptr时,引用计数就会减少。只有当引用计数降到0时,才会删除原生指针。类似的还要处理若引用。此外,引用计数还要保证线程安全。
用法举例
参考测试项目代码ModernCppTest/modrenc_smart_pointer.cpp
主要包含:
- 共享指针用法&声明方式
- 独占指针用法&声明方式
- 弱指针的用法
#include "ModernCppTestHeader.h"
#include <memory>
#define LOG_UNIQUE_PTR_VALID(ptr) if(ptr) LOG(#ptr << " unique_ptr valid"); else LOG(#ptr << " unique_ptr invalid")
#define LOG_WEAK_PTR_IF_VALID(ptr, exp) if(auto tp = p2.lock()) exp else LOG(#ptr << " weak_ptr invalid")
namespace n_smart_pointer {
class Obj {
public:
Obj(int ID) : ID(ID) { LOG("Obj addr " << this << " ID " << ID << " Create"); }
~Obj() { LOG("Obj addr " << this << " ID " << ID << " Release"); }
int ID;
};
class Item {};
}
using LocaObj = n_smart_pointer::Obj;
void smart_pointer_test()
{
LOG_FUNC();
LOG_TAG("std::shared_ptr 共享指针");
{
{
std::shared_ptr<LocaObj> p1 = std::make_shared<LocaObj>(1);
{
std::shared_ptr<LocaObj> p2 = p1;
LOG_VAR_DESC(p1->ID, " p1->ID");
LOG_VAR_DESC(p2->ID, " p2->ID");
LOG("p2 离开作用域");
}
LOG("p1 离开作用域");
}
}
LOG_TAG("std::shared_ptr 共享指针的声明方式");
{
std::shared_ptr<LocaObj> p1(new LocaObj(1));
std::shared_ptr<LocaObj> p2 = std::make_shared<LocaObj>(2);
std::shared_ptr<LocaObj> p3(p2);
}
LOG_TAG("std::unique_ptr 唯一指针");
{
{
std::unique_ptr<LocaObj> p1;
{
std::unique_ptr<LocaObj> p2 = std::make_unique<LocaObj>(1);
LOG_VAR_DESC(p2->ID, " p2->ID");
LOG("唯一指针转移控制权使用=运算符报错,须使用std::move()");
// std::unique_ptr<LocaObj> p1 = p2;
p1 = std::move(p2);
LOG_VAR_DESC(p1->ID, " p1->ID");
LOG("测试被std::move的p2是否还有效,可以看到 无效");
LOG_UNIQUE_PTR_VALID(p2);
LOG("p2 离开作用域, ID为1的Obj没有析构");
}
LOG("p1 离开作用域, ID为1的Obj析构");
}
}
LOG_TAG("std::unique_ptr 唯一指针的声明方式");
{
std::unique_ptr<LocaObj> p1(new LocaObj(1));
std::unique_ptr<LocaObj> p2 = std::make_unique<LocaObj>(2);
std::unique_ptr<LocaObj> p3(std::move(p2));
}
LOG_TAG("std::weak_ptr 弱指针");
{
std::shared_ptr<LocaObj> p1 = std::make_shared<LocaObj>(1);
std::weak_ptr<LocaObj> p2 = p1;
LOG_VAR_DESC(p1->ID, " p1->ID");
LOG_WEAK_PTR_IF_VALID(p2, LOG_VAR_DESC(tp->ID, " p2->ID"););
LOG("p1 reset Obj对象被释放");
p1.reset();
LOG("p2 弱指针失效");
LOG_WEAK_PTR_IF_VALID(p2, LOG_VAR_DESC(tp->ID, " p2->ID"););
}
}
现代C++(Modern C++)基本用法实践:五、智能指针的更多相关文章
- 必须要注意的 C++ 动态内存资源管理(五)——智能指针陷阱
必须要注意的 C++ 动态内存资源管理(五)——智能指针陷阱 十三.小心使用智能指针. 在前面几节已经很详细了介绍了智能指针适用方式.看起来,似乎智能指针很强大,能够很方便很安全的管理 ...
- Linux中sed的用法实践
Linux中sed的用法实践 参考资料:https://www.cnblogs.com/emanlee/archive/2013/09/07/3307642.html http://www.fn139 ...
- 20145203盖泽双《网络对抗技术》实践五:MSF基础应用
20145203盖泽双<网络对抗技术>实践五:MSF基础应用 1.实践目标 掌握metasploit的基本应用方式,掌握常用的三种攻击方式的思路.下面是我自己做的时候用的四个套路. (1) ...
- Linux及安全实践五——字符集编码
Linux及安全实践五——字符集编码 一.ASCII码 在表中查找出英文字母LXQ相对应的十六进制数值为: 4c 58 51 在终端中输入命令:vim test1.txt 在vim页面输入命令:%!x ...
- nodejs 实践:express 最佳实践(五) connect解析
nodejs 实践:express 最佳实践(五) connect解析 nodejs 发展很快,从 npm 上面的包托管数量就可以看出来.不过从另一方面来看,也是反映了 nodejs 的基础不稳固,需 ...
- 智能指针shared_ptr的用法
为了解决C++内存泄漏的问题,C++11引入了智能指针(Smart Pointer). 智能指针的原理是,接受一个申请好的内存地址,构造一个保存在栈上的智能指针对象,当程序退出栈的作用域范围后,由于栈 ...
- 智能指针unique_ptr的用法
unique_ptr是独占型的智能指针,它不允许其他的智能指针共享其内部的指针,不允许通过赋值将一个unique_ptr赋值给另一个unique_ptr,如下面错误用法: std::unique_pt ...
- C++ template的一些高级用法(元编码,可变参数,仿函数,using使用方法,. C++ 智能指针)
1 . 通用函数可变参数模板 对于有些时候,我们无法确切的知道,函数的参数个数时,而又不想过多的使用所谓的函数重载,那么就可以效仿下面的例子: #include<iostream> #i ...
- c++ 智能指针用法详解
本文介绍c++里面的四个智能指针: auto_ptr, shared_ptr, weak_ptr, unique_ptr 其中后三个是c++11支持,并且第一个已经被c++11弃用. 为什么要使用智能 ...
- C++ auto_ptr智能指针的用法
C++中指针申请和释放内存通常采用的方式是new和delete.然而标准C++中还有一个强大的模版类就是auto_ptr,它可以在你不用的时候自动帮你释放内存.下面简单说一下用法. 用法一: std: ...
随机推荐
- Java的对象克隆
本节我们会讨论 Cloneable 接口,这个接口指示一个类提供了一个安全的 clone() 方法. Object 类提供的 clone() 方法是 "浅拷贝",并没有克隆对象中引 ...
- SQL server数据库拼接语句(STUFF)用法
我对SQLserver 中STUFF函数的理解是在sql server中将字符串中的第一个字符串某一部分字符替换成另外一部分,组成新的字符串数据. STUFF(character_expression ...
- iOS APP启动广告实现方式 与 APP唤端调用
APP启动广告功能实现要从2个方面思考 一是UI方案,怎样处理广告页与主页之间的切换方式. 二是广告页展示时机,是使用后台实时广告数据还是使用本地缓存广告数据.后台数据方式获取广告最新但是用户要等待后 ...
- ChatGPT会对我们日常生活带来什么影响?这些技术会改变我们学习阅读工作方式吗?
ChatGPT会对我们日常生活带来什么影响?这些技术会改变我们学习阅读工作方式吗? AI 这个话题很火,我也一直在关注着,很多人甚至觉得 AI 会改变世界,也许你会好奇:ChatGPT 会在三年内终结 ...
- 2022-08-18:每一个序列都是[a,b]的形式,a < b 序列连接的方式为,前一个序列的b,要等于后一个序列的a 比如 : [3, 7]、[7, 13]、[13, 26]这三个序列就可以依次连
2022-08-18:每一个序列都是[a,b]的形式,a < b 序列连接的方式为,前一个序列的b,要等于后一个序列的a 比如 : [3, 7].[7, 13].[13, 26]这三个序列就可以 ...
- 2022-04-21:给定一个包含 [0,n) 中不重复整数的黑名单 blacklist, 写一个函数从 [0, n) 中返回一个不在 blacklist 中的随机整数, 对它进行优化使其尽量少调用系
2022-04-21:给定一个包含 [0,n) 中不重复整数的黑名单 blacklist, 写一个函数从 [0, n) 中返回一个不在 blacklist 中的随机整数, 对它进行优化使其尽量少调用系 ...
- vue全家桶进阶之路7:Vue的第一个程序
<!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8" ...
- aggregate和annotate⽅法
现在来看下⼏组实际使⽤案例.使⽤前别忘了import Avg, Max, Min或者Sum⽅法哦from django.db.models import Avg, Max, Min计算学⽣平均年龄, ...
- airasia Superapp × HMS Core:便捷出行,悦享全程
2023年5月9日-5月11日,HUAWEI P60系列及旗舰产品发布会在欧洲德国.中东非阿联酋.亚太马来西亚.拉美墨西哥陆续举办,为消费者带来高端影像旗舰HUAWEI P60 Pro及系列全场景智能 ...
- ✗ CocoaPods not installed.
mac 配置 flutter 会提示许多 关于xcode的 如图 显示 ✗ CocoaPods installed but not initialized. 其实最开始提示的是 ✗ CocoaPods ...