1.新的标准库提供了两种智能指针类型,shared_ptr允许多个指针指向同一个对象,unique_ptr则独占所指的对象。标准库还定义了一个名为weak_ptr的伴随类,它是一种弱引用,指向shared_ptr所管理的对象。

2.智能指针也是模板,默认初始化的智能指针中保存着一个空指针。

3.智能指针的操作:

4.make_shared用参数来构造指定类型的对象。

std::shared_ptr<int> p = std::make_shared<int>(42);    // 指向一个值为42的int型shared_ptr
std::shared_ptr<std::string> p1 = std::make_shared<std::string>(2,'1'); // 指向一个值为"11"的string类型shared_ptr
std::shared_ptr<int> p2 = std::make_shared<int>(); // 值初始化为0

5.我们可以认为每个shared_ptr有一个引用计数,无论何时我们拷贝一个shared_ptr,计数器都会递增,当我们给shared_ptr赋予一个新值或是shared_ptr被销毁时,计数器就会递减。

6.当指向一个对象的最后一个shared_ptr被销毁时,shared_ptr类会自动销毁此对象。它是通过析构函数完成销毁工作的。所占用的内存也会被释放。

7.shared_ptr在无用之后仍然保留的一种可能情况是,你将shared_ptr放在一个容器中(其实已经做了拷贝),随后重排了容器,从而不再需要某些元素,而只使用其中一部分,所以要记得用erase删除不再需要的那些元素。

8.在自由空间分配的内存是无名的,因此new无法为其分配的对象命名,而是返回一个指向该对象的指针。

9.默认情况下,动态分配的对象是默认初始化的。这意味着内置类型或组合类型的值将是为定义的,而类类型的对象将用默认构造函数进行初始化。也可以对动态分配的内存进行值初始化,只需在类型名之后跟一对空括号即可。

int *p = new int;    // 默认初始化,*p的值未定义
int *p1 = new int(); // 值初始化为0,*p1为0

10.当我们提供一个括号包围的初始化器,就可以使用auto从此初始化器来推断我们想要分配的对象的类型,但是正因如此,只有当括号中仅有单一初始化器时才可以使用auto。

int a, b;
int i = ;
auto p = new auto(i); // p指向一个值为1的int型,但不是指向i
auto p1 = new auto{a,b}; // 错误,括号中只能有单个初始化器

11.定位new表达式允许我们向new传递额外的参数。

int *p = new int;    // 如果分配失败,new抛出std::bad_alloc
int *p1 = new (std::nothrow) int; // 如果分配失败,new返回一个空指针

12.我们传递给idelete的指针必须指向动态分配的内存,或者是一个空指针。释放一块并非new分配的内存,或者将相同的指针值释放多次,其行为是未定义的。

13.我们可以用new返回的指针来初始化智能指针,接受指针参数的智能指针构造函数是explicit的,因此不能将一个内置指针隐式转换为一个只能指针,必须使用直接初始化形式来初始化一个智能指针。

std::shared_ptr<int> p = new int();     // 错误,必须使用直接初始化形式
std::shared_ptr<int> p1(new int()); // 正确,使用了直接初始化形式

14.默认情况下,一个用来初始化智能指针的普通指针必须指向动态内存,因为智能指针默认使用delete释放它所关联的对象,我们可以将智能指针绑定到一个指向其他类型的资源的指针上,但是为了这样做,必须提供自己的操作来代替delete。

15.定义和改变shared_ptr的其他方法:

16.使用一个内置指针来访问一个智能指针所负责的对象是很危险的,因为我们无法知道对象何时会被销毁。

void test(std::shared_ptr<int> p)
{
// ...
} int *p = new int(); // p是一个普通指针,不是一个智能指针
test(p); // 错误,不能将int*转换为一个shared_ptr<int>
test(std::shared_ptr<int>(p)); // 合法的,但是一旦函数调用结束,p指向的内存就被释放了
int j = *p; // 未定义的,p是一个空悬指针

17.get用来将指针的访问权限传递给代码,你只有在确定代码不会delete指针的情况下,才能使用get。特别的,永远不要用get初始化另一个智能指针或者为另一个智能指针赋值。

std::shared_ptr<int> p(new int());    // p的引用计数为1
int *q = p.get(); // 正确,但是注意不要让q被释放
if ()
{
std::shared_ptr<int> p1(q);
} // 程序块结束,p1被销毁,所指的内存被释放
int i = *p; //未定义,p所指的内存被释放了

18.与赋值类似,reset会更新引用计数,如果需要的话,会释放p指向的对象。reset经常与unique一起使用。

std::shared_ptr<int> p(new int());
// ...
if (!p.unique())
p.reset(new int()); // 我们不是唯一用户,分配新的拷贝
else
(*p)++; // 我们是唯一用户,可以直接改变对象的值

19.如果在new和delete之间发生异常,且异常未被捕获,则内存就永远不会被释放,使用智能指针就可以避免这个问题。

20.当一个shared_pt(unique_ptr也一样)r被销毁时,默认会对它管理的指针进行delete操作,我们可以自己定义一个删除器函数来代替delete操作。

class test
{
public:
~test() {};
}; void my_delete(test *t)
{
t->~test();
} std::shared_ptr<test> p(new test(), my_delete); // p被销毁时会调用my_delete

21.unique_ptr也必须采用直接初始化的形式,而且unique_ptr不支持普通的拷贝或赋值操作,但是有如下操作。

  • 如果release返回的指针没有被其他的指针保存,则不仅内存没有被释放,而且我们丢失了指针

22.不能拷贝unique_ptr的规则有一个例外,我们可以拷贝或赋值一个将要被销毁的unique_ptr。

std::unique_ptr<int> clone(int i)
{
return std::unique_ptr<int>(new int(i));
}
// or
std::unique_ptr<int> clone1(int i)
{
std::unique_ptr<int> ret(new int(i));
return ret;
}

23.weak_ptr是一种不控制所指向对象生存期的智能指针,它指向一个shared_ptr管理的对象。所以weak_ptr不会改变shared_ptr的引用计数。

24.weak_ptr定义了如下操作:

25.由于对象可能不存在,我们不能使用weak_ptr直接访问对象,而必须调用lock.

std::shared_ptr<int> p(new int());
std::weak_ptr<int> p1(p); if (std::shared_ptr<int> np = p1.lock()) // 如果np为空则不可操作
{
(*np)++;
}

C++ Primer 笔记——智能指针的更多相关文章

  1. c++学习笔记——智能指针

    智能指针是为了便于管理动态内存,能够自动管理释放所指向的对象. 智能指针共有三种:1.shared_ptr允许多个指针指向同一个对象:2.unique_ptr独占所指向的对象:3.weak_ptr是一 ...

  2. C++智能指针初学小结

    本篇随笔仅作为个人学习<C++ Primer>智能指针一节后的部分小结,抄书严重,伴随个人理解.主要介绍shared_ptr.make_shared.weak_ptr的用法和联系. C++ ...

  3. C++中的智能指针、轻量级指针、强弱指针学习笔记

    一.智能指针学习总结 1.一个非const引用无法指向一个临时变量,但是const引用是可以的! 2.C++中的delete和C中的free()类似,delete NULL不会报"doubl ...

  4. C++笔记(11) 智能指针

    1. 设计思想 智能指针是行为类似于指针的类对象,但这种对象还有其他功能.首先,看下面的函数: void remodel(std::string & str) { std::string * ...

  5. C++的智能指针学习笔记(初)

    C++ primer plus 16.2节介绍了auto_ptr,该模板类在C++11中已弃用,目前已被shared_ptr代替. auto_ptr又叫做智能指针,用于管理动态内存分配的用法. 为什么 ...

  6. EC笔记:第三部分:17、使用独立的语句将newed对象放入智能指针

    一般的智能指针都是通过一个普通指针来初始化,所以很容易写出以下的代码: #include <iostream> using namespace std; int func1(){ //返回 ...

  7. C++11智能指针读书笔记;

    智能指针是一个类对象,而非一个指针对象. 原始指针:通过new建立的*指针 智能指针:通过智能指针关键字(unique_ptr, shared_ptr ,weak_ptr)建立的指针 它的一种通用实现 ...

  8. C++ Primer : 第十二章 : 动态内存之shared_ptr与new的结合使用、智能指针异常

    shared_ptr和new结合使用 一个shared_ptr默认初始化为一个空指针.我们也可以使用new返回的指针来初始化一个shared_ptr: shared_ptr<double> ...

  9. osg(OpenSceneGraph)学习笔记1:智能指针osg::ref_ptr<>

    OSG的智能指针,osg::ref_ptr<> osg::Referenced类管理引用计数内存块,osg::ref_ptr需要使用以它为基类的其它类作为模板参数. osg::ref_pt ...

随机推荐

  1. Light oj 1021 - Painful Bases

    题意:  给一个B进制的数,一个10进制的数K,B进制数有x位, 对着x位进行全排列的话,有x!种可能, 问这x!的可能中,有多少种可以整除K,各个位置上的数字都不同. 思路:状态压缩,数位DP #i ...

  2. light oj 1254 - Prison Break 最短路

    题目大意:n个点m条边的有向图,q次询问c,s,t,表示汽车邮箱容量为c,求从起点s到终点t的最小费用.汽车在每个点可以加任意的油,每个点的单位油价为a[i]. 题目思路:利用最小费优先队列优化最短路 ...

  3. Light oj 1281 - New Traffic System 多状态最短路

    题目大意:有向图,新计划的地铁,有k个计划新路,利用现有的铁路.k条新路和限定只能用d条新路,找出从0到n-1的最短路径 题目思路:用dist[u][use],储存使用use条新路,到达节点u的最短路 ...

  4. 【NLP CS224N笔记】Lecture 3 GloVe: Global Vectors for Word Representation

    I. 复习word2vec的核心思路 1. Skip-gram 模型示意图: 2.word vectors的随机梯度 假设语料库中有这样一行句子: I love deep learning and N ...

  5. 什么是openstack

    什么是openstack OpenStack是一个云平台管理的项目,它不是一个软件. 这个项目由几个主要的组件组合起来完成一些具体的工作. 想要了解openstack,第一步我们可以观察他的概念图: ...

  6. nodejs -Promise

    创建一个 readFile.js,读取三个文件abc的内容并输出到控制台 var fs = require('fs') fs.readFile('./a.txt','utf-8',function ( ...

  7. 【ARTS】01_17_左耳听风-20190304~20190310

    ARTS: Algrothm: leetcode算法题目 Review: 阅读并且点评一篇英文技术文章 Tip/Techni: 学习一个技术技巧 Share: 分享一篇有观点和思考的技术文章 Algo ...

  8. Windows Server 2008 R2忘记密码导致无法修改密码

    其实有个记住密码的工具还是简单的,登陆系统后,CMD输入命令: net user administrator NewPassword 新密码即时生效

  9. 两种 AuthorizationSchemes 在 ASP.NET Core 2

    Welcome to IdentityServer4:  https://identityserver4.readthedocs.io/en/release/   支持 ASP.NET Core 2 ...

  10. boost--asio--读写大总结

    NO.1 ASIO 读操作大总结: A. Boos::asio::read 同步读方式 void client::read_data(char   * sourse  , int num ) { bo ...