首页 > 编程语言 >c++ primer plus 第16章string 类和标准模板库,16.2.2 有关智能指针的注意事项

c++ primer plus 第16章string 类和标准模板库,16.2.2 有关智能指针的注意事项

时间:2024-07-18 19:55:04浏览次数:16  
标签:string 16 auto c++ 智能 ptr pwin 指针

c++ primer plus 第16章string 类和标准模板库,16.2.2 有关智能指针的注意事项

c++ primer plus 第16章string 类和标准模板库,16.2.2 有关智能指针的注意事项

文章目录


16.2.2 有关智能指针的注意事项

为何有三种智能指针呢?实际上有4种,但本书不讨论weakptr。为何摒弃 auto ptr 呢?先来看下面的赋值语句:

auto ptr<string>ps (new string("I reiqned lonely as a cloud."));
auto ptr<string>vocation;
vocation =psi

上述赋值语句将完成什么工作呢?如果ps和 vocation 是常规指针,则两个指针将指向同一个string 对象。这是不能接受的,因为程序将试图删除同一个对象两次–一次是ps过期时,另一次是 vocation 过期时。要避免这种问题,方法有多种。
定义赋值运算符,使之执行深复制。这样两个指针将指向不同的对象,其中的一个对象是另一个对象的副本。
建立所有权(ownership)概念,对于特定的对象,只能有一个智能指针可拥有它,这样只有拥有对象的智能指针的构造函数会删除该对象。然后,让赋值操作转让所有权。这就是用于autoptr和 unique ptr 的策略,但 unique ptr 的策略更严格。
创建智能更高的指针,跟踪引用特定对象的智能指针数。这称为引用计数(referencecounting)。
例如,赋值时,计数将加1,而指针过期时,计数将减1。仅当最后一个指针过期时,才调用 delete。
这是 shared ptr 采用的策略。
当然,同样的策略也适用于复制构造函数。
每种方法都有其用途。程序清单16.6是一个不适合使用autoptr的示例。

程序清单16.6 fowl.cpp

// fowl.cpp  -- auto_ptr a poor choice
#include <iostream>
#include <string>
#include <memory>

int main()
{
    using namespace std;
    auto_ptr<string> films[5] =
    {
        auto_ptr<string> (new string("Fowl Balls")),
        auto_ptr<string> (new string("Duck Walks")),
        auto_ptr<string> (new string("Chicken Runs")),
        auto_ptr<string> (new string("Turkey Errors")),
        auto_ptr<string> (new string("Goose Eggs"))
    };
    auto_ptr<string> pwin;
    pwin = films[2];   // films[2] loses ownership

    cout << "The nominees for best avian baseball film are\n";
    for (int i = 0; i < 5; i++)
        cout << *films[i] << endl;
    cout << "The winner is " << *pwin << "!\n";
    // cin.get();
    return 0;
}

消息 core dumped 表明,错误地使用 auto pt 可能导致问题(这种代码的行为是不确定的,其行为可能随系统而异)。这里的问题在于,下面的语句将所有权从lms[2]转让给 pwin:

这导致 fims[2]不再引用该字符串。在 auto ptr 放弃对象的所有权后,便可能使用它来访问该对象。当程序打印 flms[2]指向的字符串时,却发现这是一个空指针,这显然讨厌的意外。
如果在程序清单 16.6中使用 shared ptr 代替 auto ptr(这要求编译器支持 C++11 新增的 shared pt类),则程序将正常运行,
差别在于程序的如下部分:

shared ptr<string> pwin;
pwin = films[2];

这次 pwin 和 flms[2]指向同一个对象,而引用计数从1增加到2。在程序末尾,后声明的 pwin 首先调用其析构函数,该析构函数将引用计数降低到1。然后,shared ptr 数组的成员被释放,对 flmsp[2]调用析
构函数时,将引用计数降低到0,并释放以前分配的空间。因此使用 shared ptr 时,程序清单16.6运行正常;而使用 auto ptr 时,该程序在运行阶段崩溃。如果使用 unique ptr,结果将如何呢?与 auto ptr 一样,unique ptr也采用所有权模型。但使用 unique ptr 时,程序不会等到运行阶段崩溃,而在编译器因下述代码行出现错误:

pwin = films[2];

显然,该进一步探索autoptr和unique ptr 之间的差别。

标签:string,16,auto,c++,智能,ptr,pwin,指针
From: https://blog.csdn.net/zhyjhacker/article/details/140531194

相关文章

  • C++ 面向对象程序设计 ---- 类2重点
    1.构造函数,代替Init()函数构造函数的特点1.函数名与类名相同2.无返回值,void也不需要写3.对象实例化时,系统会自动调用构造函数4.构造函数可以重载classDate{public://函数名与类名相同,无返回值Date()//函数重载,无参{_year=1;......
  • 什么是信息指纹和信息加密——《数学之美》第16、17章以及其他各种资料的读书笔记
    目录1.信息指纹1.1概念1.2相关算法的演进历程1.3 哈希碰撞1.4 雪崩效应1.5 应用场景2.信息加密2.1密码学的简要历史2.1.1古代密码学:智慧的萌芽2.1.2 中世纪至文艺复兴:密码术的兴起2.1.3 近代密码学:机械密码机的诞生2.1.4 现代密码学:复杂科学的诞生2.......
  • 16 触发器
    概念触发器trigger是一种特殊类型的存储过程。触发器主要是通过事件进行触发而被执行,而存储过程是可以直接通过存储过程的名字被调用执行触发器的作用1、可以在写入数据表前,强制检验或转换数据,以保证数据安全2、触发器发生错误时,异动的结果会被撤销如果触发器执行发生错误......
  • OI学习笔记(C++)
    笔记完整版链接参照oi.wiki整理了基础dp的一些笔记:学习笔记+模板(Adorable_hly)(自己结合网络和做题经验总结的,dalao勿喷)第一大板块:DP动态规划适用场景:1.最优化原理:若该问题所包含的子问题解是最优的,就称该问题具有最优子结构,满足最优化原理。2.无后效性:指某一阶段的状......
  • Error: Expected ref to be a function, a string, an object returned by React.crea
    1、完整报错:Error:Expectedreftobeafunction,astring,anobjectreturnedbyReact.createRef(),ornull.atcoerceRef(react-dom.development.js:14873:1)atreconcileSingleElement(react-dom.development.js:15723:1)atreconcileChildFibers(re......
  • C++异常
    异常描述std::exception该异常是所有标准C++异常的父类。std::bad_alloc该异常可以通过new抛出。std::bad_cast该异常可以通过dynamic_cast抛出。std::bad_typeid该异常可以通过typeid抛出。std::bad_exception这在处理C++程序中无法预期的异......
  • C++文件和流
    要在C++中进行文件处理,必须在C++源代码文件中包含头文件<iostream>和<fstream>。数据类型描述fstream该数据类型通常表示文件流,且同时具有ofstream和ifstream两种功能,这意味着它可以创建文件,向文件写入信息,从文件读取信息。ofstream该数据类型表示输出......
  • C++中的vector对应Java中的什么类型?
    C++中的vector对应Java中的ArrayList类型。‌C++中的vector和Java中的ArrayList都是可变长的数组或数组列表,‌它们具有以下相似特性:‌两者都是动态数组,‌可以根据需要自动增长。‌它们都支持通过索引访问元素,‌并且元素是有序的。‌它们都提供了添加、‌删除和查询元素的方法......
  • 7.16 网络流
    网络流和费用流,其实知道如何建图之后就可以直接套板子了,但正如konata所言,这些题如果不是在今天作业里,打死也想不到要用网络流。善意的投票这个是最小割的典型问题,把睡午觉视为集合a,即原点,不睡视为集合b,即汇点,那么对每个点,连向自己意愿的容量为1,违背的容量为0。对于一对朋友,他们......
  • 基于SpringBoot的宠物领养系统-07863(免费领源码+开发文档)可做计算机毕业设计JAVA、PHP
    摘 要21世纪的今天,随着社会的不断发展与进步,人们对于信息科学化的认识,已由低层次向高层次发展,由原来的感性认识向理性认识提高,管理工作的重要性已逐渐被人们所认识,科学化的管理,使信息存储达到准确、快速、完善,并能提高工作管理效率,促进其发展。论文主要是对宠物领养系统......