C++11的一些新特性|右值引用|STL中的一些变化

news2025/1/14 0:45:49

文章目录

1、{}初始化

2、声明

2.1auto

2.2、decltype

2.3、nullptr

2.4.范围for循环

3、STL中的一些新变化

3.1.新容器

3.2容器中的一些新方法

4.右值引用和移动语义

左值引用和右值引用

左值引用的短板:

右值引用使用场景和意义:

move的作用:

完美转发

5.新的类功能

移动构造和移动赋值

类成员变量初始化

强制生成默认函数关键字default:

禁止生成默认函数的关键字delete



1、{}初始化

在c++98中,标准允许使用{}对数组或者结构体进行初始化,如下:

struct Point
{
 int _x;
 int _y;
};

int array1[] = { 1, 2, 3, 4, 5 };
int array2[5] = { 0 };
Point p = { 1, 2 };

c++11扩大了{}的使用范围,使其可用于对所有内置类型和用户自定义类型,使用初始化列表时,可添加等号,也可不添加

struct Point
{
 int _x;
 int _y;
};
int main()
{
 int x1 = 1;
 int x2{ 2 };
 int array1[]{ 1, 2, 3, 4, 5 };
 int array2[5]{ 0 };
 Point p{ 1, 2 };
 // C++11中列表初始化也可以适用于new表达式中
 int* pa = new int[4]{ 0 };
 return 0;
}

//创建对象时,也可以使用列表初始化方式调用构造函数初始化
int main()
{
 Date d1(2022, 1, 1); // old style
 // C++11支持的列表初始化,这里会调用构造函数初始化
 Date d2{ 2022, 1, 2 };
 Date d3 = { 2022, 1, 3 };
 return 0;
}

2、声明

2.1auto

C++98auto是一个存储类型的说明符,表明变量是局部自动存储类型,但是局部域中定义局部的变量默认就是自动存储类型,所以auto就没什么价值了。C++11中废弃auto原来的用法,将其用于实现自动类型推断。这样要求必须进行显示初始化,让编译器将定义对象的类型设置为初始化值的类型

int i = 10;
auto p = &i;

2.2、decltype

关键字decltype将变量的类型声明为表达式指定的类型

const int x = 1;
double y = 2.2;
decltype(x * y) ret; // ret的类型是double
decltype(&x) p;      // p的类型是int*

2.3、nullptr

c++中Null被定义为0,这样可能会带来一些问题,因为0既能指针常量,又能表示整形常量。c++中nullptr用来表示空指针。

2.4.范围for循环

之前已经用过很多次,这里不再详细阐述。

3、STL中的一些新变化

3.1.新容器

3.2容器中的一些新方法

       比如提供了cbegin和cend方法返回const迭代器等。但是实际意义不大,因为begin和end也是可以返回const迭代器的。实际上c++11更新后,容器中新增的新方法最后用的插入接口的右值引用版本。接下来先介绍右值引用。

4.右值引用和移动语义

左值引用和右值引用

传统的c++语法中就有引用的语法,而c++11中新增了右值引用的语法特性,无论是左值引用还是右值引用,都是给对象取别名。

左值:左值是一个数据的表达(如变量名或者解引用的指针),我们可以获取它的地址,也可以对他进行赋值。左值可以出现在赋值符号的左边右值不能出现在赋值符号左边。定义时const修饰后的左值,不能给他赋值,但是可以获取他的地址。左值引用就是给左值的引用,给左值取别名。

int * p = new int(0);
int b = 1;
const int c = 2;

//对左值引用
int *&rp = p;
int &rb= b;
const int& rc = c;
int& pvalue = *p;

右值:右值也是一个数据表达式,如:字面变量表达式返回值函数返回值(临时变量)(这个不能是左值引用返回)等。右值可以出现在赋值符号的右边,但是不能出现在赋值符号的左边,右值不能取地址。右值引用就是对右值的引用,给右值取别名。

需要注意的是,右值是不能取地址。但给右值取别名后,会导致右值被存储到特定位置,且可以取到该位置地址。例如:不能取字面量10的地址,但是rr1引用后,可以对rr1取地址,也可以修改rr1,如果不想rr1被修改,可以用const int && rr1取引用。

double x = 1.1 , y = 2.2;

//常见的右值
10
x+y;
fmin(x,y)


//以下几个都是对右值的右值引用
int&& rr1 = 10;
double&& rr2 = x+y;
double & rr3 = fmin(x+y);



//这里编译会报错,error c2106 "=":左操作数必须为左值
10 = 1;
x+y = 1;
fmin(x,y) = 1;

左值引用和右值引用比较:

1.左值引用只能引用左值,不能引用右值

2.const左值引用既可以引用左值,也可以引用右值

右值引用总结:

1.右值引用只能引用右值,不能引用左值

2.但是右值引用可以引用move以后的左值

int && rr1 = 10;

 // error C2440: “初始化”: 无法从“int”转换为“int &&”
 // message : 无法将左值绑定到右值引用
 int a = 10;
 int&& r2 = a;

//右值引用可以引用move以后的左值
int &&rr3 = std::move(a);

左值引用的短板:

当函数返回一个局部变量,出了作用域就不存在了,就不能使用左值引用返回,只能传值返回,传值返回至少会导致一次拷贝构造(旧一些的编译器可能会两次拷贝构造)

右值引用使用场景和意义:

1.作为返回值和参数都可以提高效率。在类中增加移动构造,移动构造本质上是将参数右值的资源窃取,占为己有,那么就不用做深拷贝,所以叫它移动构造,就是窃取别人的资源来构造自己

// 移动构造
string(string&& s)
 :_str(nullptr)
 ,_size(0)
 ,_capacity(0)
{
 cout << "string(string&& s) -- 移动语义" << endl;
 swap(s);
}
int main()
{
 bit::string ret2 = bit::to_string(-1234);
 return 0;
}

2.移动赋值  类中增加移动赋值函数

// 移动赋值
string& operator=(string&& s)
{
cout << "string& operator=(string&& s) -- 移动语义" << endl;
swap(s);
return *this;
}
int main()
{
 bit::string ret1;
 ret1 = bit::to_string(1234);
 return 0;
}
// 运行结果:
// string(string&& s) -- 移动语义
// string& operator=(string&& s) -- 移动语义

move的作用:

有些场景下,可能真的需要用右值去引用左值实现移动语义。当需要用右值引用引用一个左值时,可以通过move函数将左值转化为右值C++11中,std::move()函数位于 头文件中,该函数名字具有迷惑性,并不搬移任何东西,唯一的功能就是将一个左值强制转化为右值引用,然后实现移动语义。

int main()
{
    string s1("hello world"); // 这里s1是左值,调用的是拷贝构造
    string s2(s1);
 // 这里我们把s1 move处理以后, 会被当成右值,调用移动构造
 // 但是这里要注意,一般是不要这样用的,因为我们会发现s1的 资源被转移给了s3,s1被置空了。
 string s3(std::move(s1));
 return 0;
}

完美转发

模板中的&&万能引用

void Fun(int &x){ cout << "左值引用" << endl; }
void Fun(const int &x){ cout << "const 左值引用" << endl; }
比特就业课
std::forward 完美转发在传参的过程中保留对象原生类型属性
void Fun(int &&x){ cout << "右值引用" << endl; }
void Fun(const int &&x){ cout << "const 右值引用" << endl; }
// 模板中的&&不代表右值引用,而是万能引用,其既能接收左值又能接收右值。
// 模板的万能引用只是提供了能够接收同时接收左值引用和右值引用的能力,
// 但是引用类型的唯一作用就是限制了接收的类型,后续使用中都退化成了左值,
// 我们希望能够在传递过程中保持它的左值或者右值的属性, 就需要用我们下面学习的完美转发

std::forward完美转发在传参的过程中保留对象原生类型属性

void Fun(int &x){ cout << "左值引用" << endl; }
void Fun(const int &x){ cout << "const 左值引用" << endl; }
void Fun(int &&x){ cout << "右值引用" << endl; }
void Fun(const int &&x){ cout << "const 右值引用" << endl; }
// std::forward<T>(t)在传参的过程中保持了t的原生类型属性。
template<typename T>
void PerfectForward(T&& t)
{
 Fun(std::forward<T>(t));
}
int main()
{
 PerfectForward(10);           // 右值
 int a;
 PerfectForward(a);            // 左值
 PerfectForward(std::move(a)); // 右值
 const int b = 8;
 PerfectForward(b);      // const 左值
 PerfectForward(std::move(b)); // const 右值
 return 0;
}

5.新的类功能

移动构造和移动赋值

原来c++类中,有6个默认成员函数:构造,析构,拷贝构造,拷贝赋值,取地址重载,const取地址重载。c++11新增了两个:移动构造函数和移动赋值运算符重载。

针对移动构造和移动赋值运算符重载,有一些需要注意的点:

如果没有自己实现移动构造,且没有实现析构,拷贝构造,拷贝赋值重载中的任意一个,那么编译器会自己生成一个默认移动构造。默认生成的移动构造,对内置类型成员会按字节拷贝,自定义类型成员,则需要看这个成员是否实现移动构造。如果实现了就调用移动构造,没有实现就调用拷贝构造。移动拷贝赋值和移动构造完全类似。如果自己提供了移动构造(移动赋值),编译器不会自动提供。

类成员变量初始化

c++11允许在类定义时给成员变量初始缺省值,默认生成构造函数会使用这些缺省值初始化

强制生成默认函数关键字default:

C++11 可以让你更好的控制要使用的默认函数。假设你要使用某个默认的函数,但是因为一些原
因这个函数没有默认生成。比如:我们提供了拷贝构造,就不会生成移动构造了,那么我们可以
使用 default 关键字显示指定移动构造生成。
class Person
{
public:
 Person(const char* name = "", int age = 0)
 :_name(name)
 , _age(age)
 {}
 Person(const Person& p)
 :_name(p._name)
 ,_age(p._age)
 {}

 Person(Person&& p) = default;
private:
 bit::string _name;
 int _age;
};
int main()
{
 Person s1;
 Person s2 = s1;
 Person s3 = std::move(s1);
 return 0;
}

禁止生成默认函数的关键字delete

如果能想要限制某些默认函数的生成,在 C++98 中,是该函数设置成 private ,并且只声明补丁
已,这样只要其他人想要调用就会报错。在 C++11 中更简单,只需在该函数声明加上 =delete
可,该语法指示编译器不生成对应函数的默认版本,称 =delete 修饰的函数为删除函数
class Person
{
public:
 Person(const char* name = "", int age = 0)
 :_name(name)
 , _age(age)
 {}
 Person(const Person& p) = delete;
private:
 bit::string _name;
 int _age;
};
int main()
{
 Person s1;
 Person s2 = s1;
 Person s3 = std::move(s1);
 return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1031982.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

干货:数据仓库基础知识(全)

1、什么是数据仓库&#xff1f; 权威定义&#xff1a;数据仓库是一个面向主题的、集成的、相对稳定的、反映历史变化的数据集合&#xff0c;用于支持管理决策。 1&#xff09;数据仓库是用于支持决策、面向分析型数据处理&#xff1b; 2&#xff09;对多个异构的数据源有效集…

【分享】Word文档如何批量转换成PDF?

Word格式比较容易编辑&#xff0c;是工作中经常用到的文档工具&#xff0c;有时候为了避免文档在传送中出现乱码&#xff0c;或者防止被随意更改&#xff0c;很多人会把Word文档转换成PDF&#xff0c;那Word文档要怎样转成PDF呢&#xff1f;如果Word文档很多&#xff0c;有没有…

两种高效的事件处理模式:Reactor模式与Proactor模式

1.Reactor模式 一般使用同步IO模型实现 &#xff08;1&#xff09;Reactor 负责监听和分发事件&#xff0c;事件类型包含连接事件、读写事件&#xff1b; &#xff08;2&#xff09;处理资源池负责处理事件&#xff0c;如 read -> 业务逻辑 -> send&#xff1b; 使用同…

three.js加载gltf文件过程以及遇到的问题

说明&#xff1a;在vue项目中使用的threejs; 刚开始&#xff0c;我是从网上下载的gltf文件&#xff0c;将.gltf 文件放在了src/assets/xxx.gltf , 对gltf格式的文件并不了解&#xff0c;使用如下方式加载gltf文件时&#xff0c; // 创建gltf加载器对象 const loader new G…

[RF学习记录][参数读取]从yaml文件读取参数变量

robotframework支持从yaml文件读取变量&#xff0c;对于比较多的参数&#xff0c;可以在yaml文件中定义好&#xff0c;在robot脚本中引用 1、定义yaml文件 文件内容如下&#xff0c;注意&#xff0c;变量和变量值之间要以4个字符分开 yaml_demo_name: tester traing_gpt_…

GC1262R/S 国产芯片,具有高效的直接 PWM 控制方式,它可以控制无刷直流电机转速,可替代APX9262R/茂达

GC1262R/S 是单线圈无刷直流电机的 电机驱动器。 具有高效的直接 PWM 控制方式&#xff0c;它可以控制无刷直流电机转速。它集成了最低速度限制模式、可调速度 斜率控制模式、软启动模式、风扇转速计、 锁保护、自动重启、TSD、OCP 和噪声控制模 式&#xff0c;噪声控制模式根据…

JavaScript的Web Worker

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ JavaScript的Web Worker⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 欢迎来到前端入门之旅&#xff01;感兴趣的可以订阅本专栏哦&#xff01;这个专栏是为那些对Web开发感兴趣、刚刚踏入前端领域的朋友们量…

全国直播基地排名前十,西南人气最旺的成都直播产业基地即将正式开园

直播产业作为新媒体行业的重要组成部分&#xff0c;在近年来得到了迅速发展。四川成都作为全国直播基地排名前十的城市之一&#xff0c;人气最旺的直播基地——成都天府蜂巢直播产业基地&#xff0c;即将正式开园。 同一决心&#xff0c;推动资源 对于即将开园的成都直播产业基…

【中秋国庆】旅行公众号文章排版素材大全

中秋国庆节长假即将来临&#xff0c;你是否已经做好了旅行计划&#xff1f;在这个举国同庆的时刻&#xff0c;何不走出家门&#xff0c;去感受大自然的壮美、领略历史的厚重以及品尝地道的美食呢&#xff1f; 随着假期的临近&#xff0c;各大公众号纷纷推出了相关文章&#xff…

拉斯克奖(Lasker Award)2023

拉斯克奖&#xff08;Lasker Award&#xff09;2023 &#x1f508;&#x1f508;&#x1f508;&#xff1a;deep learning的两位科学家获得了拉斯克奖&#xff0c;这让人不禁对今年的诺贝奖展开大胆的预测。 1. 拉斯克奖&#xff08;Lasker Award&#xff09;简介 Lasker-De…

汽配制造问题以及MES管理系统解决方案

在汽车工业中&#xff0c;零部件制造与整车制造有着显著的不同。这些差异导致了零部件制造的复杂性和高要求&#xff0c;使其成为一个高度综合的最终产品。本文将详细介绍这些差异以及针对这些差异的解决方案。 一、行业特点决定需求 汽车配件制造与整车制造存在较大不同。在整…

【李沐深度学习笔记】线性代数

课程地址和说明 线性代数p1 本系列文章是我学习李沐老师深度学习系列课程的学习笔记&#xff0c;可能会对李沐老师上课没讲到的进行补充。 线性代数 标量 标量&#xff08;scalar&#xff09;&#xff0c;亦称“无向量”。有些物理量&#xff0c;只具有数值大小&#xff0c…

为什么定时发朋友圈会更有效呢?

这是因为在同一时段 发送的好友朋友圈 无法有效分散用户的注意力 导致曝光度难以提升 而通过推广定时发朋友圈 可根据自己的粉丝活跃度 设置发圈时间 让每一条朋友圈都能高效 传递到更多的好友手中 这样&#xff0c;曝光度自然而然地就大大提升了&#xff01; 1.多个号…

SpringBoot_快速入门

一、SpringBoot 概述 文档下载地址&#xff1a;https://docs.spring.io/spring-boot/docs/current/reference/ 1.SpringBoot介绍 概述&#xff1a;SpringBoot 开发团队认为Spring操作太繁琐了&#xff0c;目的在于简化开发配置&#xff0c;能够快速搭建开发环境&#xff0c;…

【JavaWeb】你这么厉害,知道RBAC权限模型和ABAC权限模型吗?

文章目录 一.RBAC 权限模型1.RBAC的组成2.RBAC模型分类2.1.基本模型RBAC02.2.角色分层模型RBAC12.3.角色限制模型RBAC22.4.统一模型RBAC3 3.RBAC0模型核心表结构3.1.表结构设计3.2.模块划分 4.基于RBAC的延展—用户组 二.ABAC权限模型&#xff08;基于Java讲解&#xff09;1.AB…

WSUS 修补程序管理的替代方法

什么是 WSUS Windows Server Update Services &#xff08;WSUS&#xff09; 是 Microsoft for Windows Server 提供的免费附加组件&#xff0c;WSUS 从Microsoft更新中下载必要的修补程序和更新&#xff0c;并将其分发到 Windows 网络中存在的 Windows 操作系统和相关Microso…

头条文章采集工具-快速获取头条文章方法

头条文章采集。在这个信息快速爆炸的时代&#xff0c;如何轻松而高效地获取头条新闻和热门文章成为了许多人的问题。而147SEO将成为您的信息导航&#xff0c;为您打开头条文章采集的大门&#xff0c;让您畅游信息海洋。 免费全自动采集发布批量管理网站工具-147SEO​www.147seo…

应用层 DNS Linux 下解析域名命令 dig 命令使用详解

查询单个域名的 DNS 信息 dig 命令最典型的用法就是查询单个主机的信息。 linuxidclinuxidc:~$ dig www.linuxidc.com; < <>> DiG 9.11.3-1Ubuntu1.5-Ubuntu < <>> www.linuxidc.com ;; global options: cmd ;; Got answer: ;; ->>HEADER< &…

浅述AI视频智能分析技术及TSINGSEE视频智能解决方案

AI视频智能分析是一种基于人工智能的技术&#xff0c;传统的视频监控方法通常需要由人工对大量视频流进行手动观察&#xff0c;而视频智能分析技术则可以详细检查视频流&#xff0c;并能及时发现异常。TSINGSEE青犀视频AI视频智能分析技术主要包括以下几个方面&#xff1a; 1&a…

指针(二)------字符指针+数组指针+指针与数组(传参)

&#x1f493;博主csdn个人主页&#xff1a;小小unicorn ⏩专栏分类&#xff1a;C语言 &#x1f69a;代码仓库&#xff1a;小小unicorn的代码仓库&#x1f69a; &#x1f339;&#x1f339;&#x1f339;关注我带你学习编程知识 前言&#xff1a; 在指针&#xff08;一&#xf…