iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++的shared_ptr与右值如何引用
  • 569
分享到

C++的shared_ptr与右值如何引用

2023-06-19 12:06:21 569人浏览 薄情痞子
摘要

今天小编给大家分享一下c++的shared_ptr与右值如何引用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1. 介绍在

今天小编给大家分享一下c++的shared_ptr与右值如何引用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

1. 介绍

在 C++ 中没有垃圾回收机制,必须自己释放分配的内存,否则就会造成内存泄露。解决这个问题最有效的方法是使用智能指针(smart pointer)。智能指针是存储指向动态分配(堆)对象指针的类,用于生存期的控制,能够确保在离开指针所在作用域时,自动地销毁动态分配的对象,防止内存泄露。智能指针的核心实现技术是引用计数,每使用它一次,内部引用计数加1,每析构一次内部的引用计数减1,减为0时,删除所指向的堆内存。

C++11 中提供了三种智能指针,使用这些智能指针时需要引用头文件:

  • std::shared_ptr:共享的智能指针

  • std::unique_ptr:独占的智能指针

  • std::weak_ptr:弱引用的智能指针,它不共享指针,不能操作资源,是用来监视 shared_ptr 的。

共享智能指针(shared_ptr)是指多个智能指针可以同时管理同一块有效的内存,共享智能指针 shared_ptr 是一个模板类,如果要进行初始化有三种方式:通过构造函数、std::make_shared 辅助函数以及 reset 方法。共享智能指针对象初始化完毕之后就指向了要管理的那块堆内存,如果想要查看当前有多少个智能指针同时管理着这块内存可以使用共享智能指针提供的一个成员函数 use_count

2. 初始化方法

2.1 通过构造函数初始化

实例

// 使用智能指针管理一块 int 型的堆内存shared_ptr<int> ptr1(new int(520));

2.2 通过拷贝和移动构造函数初始化

调用拷贝构造函数

shared_ptr<int> ptr2(ptr1);

调用移动构造函数

std::shared_ptr<int> ptr5 = std::move(ptr2);

如果使用拷贝的方式初始化共享智能指针对象,这两个对象会同时管理同一块堆内存,堆内存对应的引用计数也会增加;
如果使用移动的方式初始智能指针对象,只是转让了内存的所有权,管理内存的对象并不会增加,因此内存的引用计数不会变化。

2.2.1 移动构造

关于移动构造,可能有些读者不太明白

移动构造是C++11标准中提供的一种新的构造方法。

在现实中有很多这样的例子,我们将钱从一个账号转移到另一个账号,将手机SIM卡转移到另一台手机,将文件从一个位置剪切到另一个位置……移动构造可以减少不必要的复制,带来性能上的提升。

我们首先来看看move函数
首先看这样一段代码

#include <iOStream>#include <cstring>#include <cstdlib>#include <vector>using namespace std;int main(){string st = "I love 进击的汪sir";vector<string> vc;vc.push_back(move(st));cout << vc[0] << endl;if (!st.empty())cout << st << endl;return 0;}

输出的结果为

C++的shared_ptr与右值如何引用

再看这样一段代码

#include <iostream>#include <cstring>#include <cstdlib>#include <vector>using namespace std;int main(){string st = "I love xing";vector<string> vc;vc.push_back(st);cout << vc[0] << endl;if (!st.empty())cout << st << endl;return 0;}

其结果为

C++的shared_ptr与右值如何引用

这两段代码唯一的不同是调用vc.push_back()将字符串插入到容器中去时,第一段代码使用了move语句,而第二段代码没有使用move语句。输出的结果差异也很明显,第一段代码中,原来的字符串st已经为空,而第二段代码中,原来的字符串st的内容没有变化。

先暂时记住这两端代码的输出结果之间的差异。
我们回到移动构造函数上

有时候我们会遇到这样一种情况,我们用对象a初始化对象b,后对象a我们就不在使用了,但是对象a的空间还在呀(在析构之前),既然拷贝构造函数,实际上就是把a对象的内容复制一份到b中,那么为什么我们不能直接使用a的空间呢?这样就避免了新的空间的分配,大大降低了构造的成本。这就是移动构造函数设计的初衷。

通俗一点的解释就是,拷贝构造函数中,对于指针,我们一定要采用深层复制,而移动构造函数中,对于指针,我们采用浅层复制。

所以在上面的例子中,如果调用移动构造函数来初始化智能指针,引用计数是不会增加的,而move函数实际上是返回的右值引用

2.2.2 右值引用

上面我们讲到了右值引用,这里就来扩展一下右值引用是啥
首先得分清楚,什么是右值,什么是左值

  • lvalue 是 loactor value 的缩写,rvalue 是 read value 的缩写

  • 左值是指存储在内存中、有明确存储地址(可取地址)的数据;

  • 右值是指可以提供数据值的数据(不可取地址);

通过描述可以看出,区分左值与右值的便捷方法是:可以对表达式取地址(&)就是左值,否则为右值 。所有有名字的变量或对象都是左值,而右值是匿名的。

C++11 中右值可以分为两种:一个是将亡值( xvalue, expiring value),另一个则是纯右值( prvalue, PureRvalue):

  • 纯右值:非引用返回的临时变量、运算表达式产生的临时变量、原始字面量和 lambda 表达式等

  • 将亡值:与右值引用相关的表达式,比如,T&& 类型函数的返回值、 std::move 的返回值等。

右值引用就是对一个右值进行引用的类型。因为右值是匿名的,所以我们只能通过引用的方式找到它。无论声明左值引用还是右值引用都必须立即进行初始化,因为引用类型本身并不拥有所绑定对象的内存,只是该对象的一个别名。通过右值引用的声明,该右值又“重获新生”,其生命周期与右值引用类型变量的生命周期一样,只要该变量还活着,该右值临时量将会一直存活下去。

右值通过&&来引用

例如:

  • int&& value = 520; 里面 520 是纯右值,value 是对字面量 520 这个右值的引用。

  • int &&a2 = a1; 中 a1 虽然写在了 = 右边,但是它仍然是一个左值,使用左值初始化一个右值引用类型是不合法的。

  • const Test& t = getObj() 这句代码的语法是正确的,常量左值引用是一个万能引用类型,它可以接受左值、右值、常量左值和常量右值。

2.3 通过 std::make_shared 初始化

通过 C++ 提供的 std::make_shared() 就可以完成内存对象的创建并将其初始化给智能指针,函数原型如下:

template< class T, class... Args >shared_ptr<T> make_shared( Args&&... args );

实例
使用智能指针管理一块 int 型的堆内存, 内部引用计数为 1

shared_ptr<int> ptr1 = make_shared<int>(520);

注意
使用 std::make_shared() 模板函数可以完成内存地址的创建,并将最终得到的内存地址传递给共享智能指针对象管理。如果申请的内存是普通类型,通过函数的()可完成地址的初始化,如果要创建一个类对象,函数的()内部需要指定构造对象需要的参数,也就是类构造函数的参数。

2.4 通过 reset 方法初始化

共享智能指针类提供的 std::shared_ptr::reset 方法函数原型如下:

void reset() noexcept;template< class Y >void reset( Y* ptr );template< class Y, class Deleter >void reset( Y* ptr, Deleter d );template< class Y, class Deleter, class Alloc >void reset( Y* ptr, Deleter d, Alloc alloc );
  • ptr:指向要取得所有权的对象的指针

  • d:指向要取得所有权的对象的指针

  • aloc:内部存储所用的分配器

实例

shared_ptr<int> ptr5;ptr5.reset(new int(250));

3. 获取原始指针

对应基础数据类型来说,通过操作智能指针和操作智能指针管理的内存效果是一样的,可以直接完成数据的读写。但是如果共享智能指针管理的是一个对象,那么就需要取出原始内存的地址再操作,可以调用共享智能指针类提供的 get () 方法得到原始地址,其函数原型如下:

T* get() const noexcept;

实例

#include <iostream>#include <string>#include <memory>using namespace std;int main(){    int len = 128;    shared_ptr<char> ptr(new char[len]);    // 得到指针的原始地址    char* add = ptr.get();    memset(add, 0, len);    strcpy(add, "博客:进击的汪sir");    cout << "string: " << add << endl;        shared_ptr<int> p(new int);    *p = 100;    cout << *p.get() << "  " << *p << endl;        return 0;}

4. 指定删除器

当智能指针管理的内存对应的引用计数变为 0 的时候,这块内存就会被智能指针析构掉了。另外,我们在初始化智能指针的时候也可以自己指定删除动作,这个删除操作对应的函数被称之为删除器,这个删除器函数本质是一个回调函数,我们只需要进行实现,其调用是由智能指针完成的。

实例

#include <iostream>#include <memory>using namespace std;// 自定义删除器函数,释放int型内存void deleteIntPtr(int* p){    delete p;    cout << "int 型内存被释放了...";}int main(){    shared_ptr<int> ptr(new int(250), deleteIntPtr);    return 0;}

删除器函数也可以是 lambda 表达式!

以上就是“C++的shared_ptr与右值如何引用”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网其他教程频道。

--结束END--

本文标题: C++的shared_ptr与右值如何引用

本文链接: https://www.lsjlt.com/news/296219.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C++的shared_ptr与右值如何引用
    今天小编给大家分享一下C++的shared_ptr与右值如何引用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1. 介绍在 ...
    99+
    2023-06-19
  • C++右值如何引用
    本篇内容介绍了“C++右值如何引用”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1.左值和右值在我们之前的文章当中,介绍的都是左值引用。C+...
    99+
    2023-06-22
  • 深入学习C++智能指针之shared_ptr与右值引用的方法
    目录1. 介绍2. 初始化方法2.1 通过构造函数初始化2.2 通过拷贝和移动构造函数初始化2.3 通过 std::make_shared 初始化2.4 通过 reset 方法初始化...
    99+
    2024-04-02
  • C语言中什么是左值引用与右值引用
    这篇文章主要介绍“C语言中什么是左值引用与右值引用”,在日常操作中,相信很多人在C语言中什么是左值引用与右值引用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C语言中什么是左值引用与右值引用”的疑惑有所帮助!...
    99+
    2023-06-16
  • C++左值与右值,右值引用,移动语义与完美转发详解
    目录C++——左值与右值、右值引用、移动语义与完美转发一、左值和右值的定义二、如何判断一个表达式是左值还是右值(大多数场景)三、C++右值引用四、std::m...
    99+
    2024-04-02
  • 如何进行C++ 11右值引用的理解
    本篇文章给大家分享的是有关如何进行C++ 11右值引用的理解,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。C++ 11中引入的一个非常重要的概念就是右值引用。理解右值引用是学习...
    99+
    2023-06-17
  • 详解C++右值引用
    目录概述移动语义(Move Semantics)完美转发(Perfect Forwarding)概述 在C++中,常量、变量或表达式一定是左值(lvalue)或右值(rvalue)。...
    99+
    2024-04-02
  • C++ 右值引用与 const 关键字详解
    C++中的const关键字的用法非常灵活,而使用const将大大改善程序的健壮性,const关键字是一种修饰符。修饰符本身,并不产生任何实际代码。就 const 修饰符而言,它用来告...
    99+
    2024-04-02
  • C++智能指针hared_ptr与右值引用的方法
    本篇内容主要讲解“C++智能指针hared_ptr与右值引用的方法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C++智能指针hared_ptr与右值引用的方法”吧!目录 介绍 初始化方法1 通...
    99+
    2023-06-20
  • C++右值引用的示例分析
    这篇文章主要介绍了C++右值引用的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。概述在C++中,常量、变量或表达式一定是左值(lvalue)或右值(rvalue)。左...
    99+
    2023-06-15
  • C++11万能引用和右值引用的方法
    这篇文章主要介绍了C++11万能引用和右值引用的方法的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C++11万能引用和右值引用的方法文章都会有所收获,下面我们一起来看看吧。正文实际上,type&&...
    99+
    2023-06-29
  • C++精要分析右值引用与完美转发的应用
    目录区分左值与右值右值引用移动语义完美转发结语区分左值与右值 在C++面试的时候,有一个看起来似乎挺简单的问题,却总可以挖出坑来,就是问:“如何区分左值与右值?&rdqu...
    99+
    2024-04-02
  • 一篇文章弄懂C++左值引用和右值引用
    目录1. 左值和右值 2. 左值引用 3. 右值引用 3.1 出现 3.2 概念 3.3 应用 3.3.1 右值引用绑定到左值上 3.3.2 std::move()本质 3.3.3 ...
    99+
    2024-04-02
  • C++11语法之右值引用的方法
    这篇文章主要讲解了“C++11语法之右值引用的方法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++11语法之右值引用的方法”吧!一、{}的扩展在原先c++的基础上,C++11扩展了很多初...
    99+
    2023-06-29
  • C++ 函数左侧值引用和右侧值引用参数的区别
    c++++中左侧和右侧值引用参数的不同之处如下:左侧值引用 (&) 指向已有对象,用于修改其状态。右侧值引用 (&&) 指向临时对象,用于获取或传递其数据。 C...
    99+
    2024-04-19
    参数 函数 引用 c++
  • C++中右值引用与移动语义的方法是什么
    今天小编给大家分享一下C++中右值引用与移动语义的方法是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。意义充分利用临时对...
    99+
    2023-07-05
  • C++右值引用与移动构造函数基础与应用详解
    目录1.右值引用1.1左值右值的纯右值将亡值右值1.2右值引用和左值引用2.移动构造函数2.1完美的移动转发1.右值引用 右值引用是 C++11 引入的与 Lambda 表达式齐名的...
    99+
    2023-02-13
    C++右值引用 C++移动构造函数
  • C++11右值引用方法是什么
    本篇内容介绍了“C++11右值引用方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!左值和右值在C++表达式的特性中有一个左值和右值的...
    99+
    2023-06-19
  • 一文带你了解C++中的右值引用与移动语义
    目录意义左值右值值类别左值纯右值将亡值左值引用右值引用std::move()移动构造&移动赋值运算符重载测试&验证意义 充分利用临时对象,避免拷贝。 左值右值 值类别...
    99+
    2023-05-13
    C++右值引用 移动语义 C++右值引用 C++ 移动语义
  • 一文搞懂C++11万能引用和右值引用
    目录前言正文万能引用结语参考:前言 我们通过一个问题来进入今天的话题:1.形如 “type&&” 的结构,就是右值引用吗?2.以下哪些属于右值...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作