C++ Primer : 第十二章 : 动态内存之shared_ptr类
在C++中,动态内存是的管理是通过一对运算符来完成的:new ,在动态内存中为对象分配空间并返回一个指向该对象的指针,delete接受一个动态对象的指针,销毁该对象,并释放该对象关联的内存。
动态内存的使用很容器出现错误,确保在正确的时间释放动态内存是极其困难的。有时候我们会忘记释放动态内存,这样就会造成内存泄露;或者是一个对象还没有使用完,就释放了它所关联的内存,再去使用它,就会造成难以预估的错误。
因此,为了更好地管理使用动态内存,C++11新标准提供了两种智能指针:shared_ptr和unique_ptr来管理动态对象。它们负责自动释放所管理的内存,shared_ptr允许多个指针同时指向同一个动态对象,unique_ptr则独占所指向的资源。标准库还定义了一个weak_ptr,它是一种弱引用,指向shared_ptr所管理的对象,但不增加shared_ptr的引用计数。三个类型都定义在 <memory>头文件中
shared_ptr类
shared_ptr<string> p1; // 指向string类型的指针
shared_ptr<int> p2; // 指向int类型的指针
默认初始化的智能指针保存着一个空指针。
| shared_ptr<T> sp | 空智能指针,可以指向类型为T的对象 |
| unique_ptr<T> up | |
| p | 将p作为一个判断条件,若p指向一个对象,则为true |
| *p | 解引用p,获得它指向的对象 |
| p->mem | 等价于(*p).mem |
| p.get() | 返回p中保存的指针 |
| swap(p, q) | 交换p和q中的指针 |
| p.swap(q) |
下面是shared_ptr独有的操作:
| make_shared<T>(args) | 返回一个shared_ptr,指向一个动态分配的类型为T的对象。使用args初始化此对象 |
| shared_ptr<T>p(q) | p是shared_ptr q的拷贝,此操作会递增q中的计数器,q中的指针必须能转换为T* |
| p = q | p和q都是shared_ptr,所保存的指针必须能相互转换,此操作会递减p的引用计数,递增q的引用计数,若p的引用计数变为0,则将其管理的原内存释放。 |
| p.unique() | 若p.use_count()为1,返回true,否则返回false |
| p.use_count() | 返回与p共享对象的智能指针数量,可能很慢,主要用于调试 |
- make_shared函数
// p3指向一个为1的int类型的shared_ptr
shared_ptr<int> p3 = make_shared<int>(1); // p4指向一个值为"9999999999"的string类型的shared_ptr
shared_ptr<string> p3 = make_shared<string>(10, '9'); // p5指向一个值初始化的int,即,值为0
shared_ptr<int> p5 = make_shared<int>();
make_shared用参数来构造给定类型的对象,例如,make_shared<string>的参数必须与string的某个构造函数的参数匹配,make_shared<int>的参数必须能初始化一个int。
如果我们不给make_shared传递任何参数,对象就会进行值初始化。
- shared_ptr的拷贝和赋值
auto p = make_shared<int>(1); // p指向的对象只有p一个引用者
auto q(p); // p和q指向相同对象,此对象有两个引用者
每个shared_ptr都有一个关联的计数器,叫做引用计数,无论何时拷贝一个shared_ptr,计数器都会递增,包括赋值,值传递的参数传递,作为返回值;当我们赋予shared_ptr
auto r= make_shared_ptr<int>(42); // r指向的int只有一个引用者
r = q; // 给r赋值,令它指向新对象
// 递增q指向的对象的引用计数
// 递减r指向的对象的引用计数
// r原来指向的对象已经没有引用者,会自动释放
- shared_ptr自动销毁所管理的对象......
- ......shared_ptr还会自动释放相关联的内存
shared_ptr<Foo> factory(T arg) {
// do some thing
// ...
return make_shared<Foo>(arg);
}
由于factory返回一个shared_ptr,所以我们可以确保它分配的对象会在恰当的时刻被释放。例如,我们可以将一个shared_ptr保存在局部变量中:
void use_factory(T arg) {
shared_ptr<Foo> p = factory(arg);
// 使用p
// p离开了作用域,它指向的内存被自动释放
}
p是use_factory的局部变量,在函数结束时它被销毁。在p被销毁时,会递减引用计数并检查是否为0。在此例中,p是唯一引用factory返回的内存的对象,由于p将要销毁,p指向的这个对象也被销毁,所占的内存也被释放。
void use_factory(T arg) {
shared_ptr<Foo> p = factory(arg);
// 使用p
return p; // 会拷贝p的一份实例,p的引用计数递增
}
这一次,use_factory在结束时,p被销毁,它指向的内存还有其他使用者,shared_ptr保证只要有任何shared_ptr对象引用它,它就不会被释放掉。
如果将一个shared_ptr存放于一个容器中,而后不再需要全部元素,而只使用其中一部分,要记得erase删除不再需要的那些元素
C++ Primer : 第十二章 : 动态内存之shared_ptr类的更多相关文章
- C++ Primer : 第十二章 : 动态内存之shared_ptr类实例:StrBlob类
StrBlob是一个管理string的类,借助标准库容器vector,以及动态内存管理类shared_ptr,我们将vector保存在动态内存里,这样就能在多个对象之间共享内存. 定义StrBlob类 ...
- C++ Primer : 第十二章 : 动态内存之shared_ptr与new的结合使用、智能指针异常
shared_ptr和new结合使用 一个shared_ptr默认初始化为一个空指针.我们也可以使用new返回的指针来初始化一个shared_ptr: shared_ptr<double> ...
- C++ Primer : 第十二章 : 动态内存之allocator类
标准库allocator类定义在头文件 <memory>中.它帮助我们将内存分配和构造分离开来,它分配的内存是原始的.未构造的. 类似vector,allocator也是一个模板类,我们在 ...
- C++ Primer : 第十二章 : 动态内存之动态内存管理(new和delete)
C++语言定义了两个运算符来分配和释放动态内存:运算符new分配内存,运算符delete释放new分配的内存. 运算符new和delete 使用new动态分配和初始化对象 在自由空间分配的内存是无名的 ...
- C++ Primer : 第十二章 : 动态内存之动态数组
动态数组的分配和释放 new和数组 C++语言和标准库提供了一次分配一个对象数组的方法,定义了另一种new表达式语法.我们需要在类型名后跟一对方括号,在其中指明要分配的对象的数目. int* arr ...
- C++ Primer : 第十二章 : 动态内存之unique_ptr和weak_ptr
unique_ptr 一个unique_ptr拥有它所管理的对象,与shared_ptr不同,unique_ptr指向的对象只能有一个用户.当unique_ptr被销毁后,它所指向的对象也被销毁. 定 ...
- C++Primer 第十二章
//1.标准库提供了两种智能指针类型来管理动态对象,均定义在头文件memory中,声明在std命名空间. // shared_ptr:允许多个指针指向同一个对象. // unique_ptr:独占所指 ...
- C++ Primer 5th 第12章 动态内存
练习12.1:在此代码的结尾,b1 和 b2 各包含多少个元素? StrBlob b1; { StrBlob b2 = {"a", "an", "th ...
- Linux内核设计与实现 总结笔记(第十二章)内存管理
内核里的内存分配不像其他地方分配内存那么容易,内核的内存分配不能简单便捷的使用,分配机制也不能太复杂. 一.页 内核把页作为内存管理的基本单位,尽管处理器最小寻址坑是是字或者字节.但是内存管理单元MM ...
随机推荐
- strstr strchr strrchr
通过函数的定义来区分: 1.strstr: 返回子串出现的第一次位置 char *strstr(const char *haystack, const char *needle) 可见,strstr函 ...
- Session初识
web服务器没有短期记忆,所以需要使用session来跟踪用户的整个会话活动.会话管理有3种解决方案: 1)使用隐藏域(很少使用) 在显示页面中使用隐藏域来保存会话ID.例如,在JSP中将input标 ...
- 用时间生成用户Id
用用户注册时的时间,作为新用户的Uid: /** * 生成用户id,用时间生成 * * @return */ public static String date2UserId() { SimpleDa ...
- BT3入门之中文语言支持
汉化: 1.更新软件库:apt-get update 2.安装中文语言包:apt-get install language-support-zh apt-get install language- ...
- c# access插入null值
c# 插入access数据库 提示错误: Parameter @DeviceLocation has no default value. 参数@DeviceLocation 的有没有默认值. Stri ...
- 有关PHP安装,基础学习
首先要安装 wamp 和 NavicatMySQLFront (要在非中文目录下) 打开DW 点击站点 ——新建站点:设置站点名称,选择本地站点文件夹:wap\www 服务器:添加 + 服务器名 ...
- 《JAVA学习笔记(14-10---14-11抽象类)》
[14-10]面向对象-抽象类的产生 /* 描述狗,行为,吼叫. 描述狼,行为,吼叫. 发现他们之间有共性,可以进行向上抽取. 当然是抽取他们的所属共性类型,犬科. 犬科这类事物,都具备吼叫行为,但是 ...
- Note_Master-Detail Application(iOS template)_05_ YJYMasterViewController.m
// YJYMasterViewController.m #import "YJYMasterViewController.h" #import "YJYDetailV ...
- linux基础命令学习五(软件包管理、下载管理)
Linux 软件包管理 本文主要是记录下RedHat系列的软件包管理. 内容分为以下二个部分:二进制包的管理,源代码包的管理 一.二进制包的管理 1.1概念 主要有RPM和YUM这两种包管理. 两 ...
- linux命令:touch
1:命令介绍: touch用来创建文件或修改文件和目录的时间戳,包括存取时间和更改时间. 2:命令格式: touch [选项] 文件 3:命令参数: -a 或--time=atime或--time ...