10 C++11
10 C++11
- 1、类型推导
- 1.1 auto关键字
- 1.2 auto类型推断本质
- 2、类型计算
- 2.1 类型计算分类
- 2.2 类型计算的四种规则
- 2.3 返回值类型计算
- 3、列表初始化
- 4、Lambda表达式
- 4.1 前置知识
- 4.2 Lambda表达式
- 4.3 捕获表
- 5、右值引用
- 5.1 概念
- 5.2 左值引用和右值引用
- 6、移动语义
1、类型推导
1.1 auto关键字
- C++98中,auto表示栈变量,通常省略不写
void foo(void){int i;auto int j;//表示在栈里分配的
}
- C++11中,给auto赋予新的语义,表示自动类型推导
- 既根据对变量进行初始化时所使用的数据的类型,由编译器自动推导出所定义变量的实际类型
auto i=0; -> int j=10;
auto j=i; -> int j=i;
1.2 auto类型推断本质
·按照定义独立对象并根据初始化数据的类型进行推导。
注意:无法自动推断const,只能自己在auto的上下文显示指明。但是有两种情况是除外的:
1:如果给出的初始化数据类型为常量指针,则可以自动推导出const
2:auto与引用的联合联用
- 按照定义独立对象并根据初始化数据的类型进行推导,所以不可能推导出引用
- 除非auto的上下文指明按照引用推导若指明按引用推导并且目标带有常属性,则可以自动推导const
/*类型推导
linux 默认是c98标准的,如果需要编译的话需要添加 -std=c++11
类型推导绝对不是类型照抄
*/
int main(){int a = 10;auto c = a;cout << "c的类型" << typeid(c).name() << endl;// typeid无法获取到对象的常属性c++;// 允许更改,说明不被const修饰cout <<"&c:"<< &c <<"&a:"<< &a << endl;const int b = 20;auto d = b;cout << "d的类型" << typeid(d).name() << endl;cout << "&d:" << &d<< "&b:" << &b << endl;d++;// 允许更改,说明不带constconst auto e = b;// 自己在auto上添加constcout << "e的类型" << typeid(e).name() << endl;cout << "&e:" << &e << "&d:" << &d << endl;// e++;// 不允许更改,说明带constauto f = &b;// 如果初始化数据的类型为常指针,则可以自动推导出constcout << "f的类型为:" << typeid(f).name() << endl;// *f = 888; // *f不允许更改f = NULL; // f是可以更改,说明推导出来的类型是 const int *return 0;
}
/*类型推导和引用的联合使用*/
int main(){int a = 10;const int b = 10;auto & d = a;cout << "d的类型" << typeid(d).name() << endl; cout << "&d:" << &d << "&a:" << &a << endl; // 地址相同,说明是别名d++;auto&e = b; // 这里指明了e是引用推导,并且b带有常属性,则可以自动推导出constcout << "e的类型" << typeid(e).name() << endl;cout << "&e:" << &e << "&b:" << &b << endl;// // 地址相同,说明是别名//e++; // 出错,说明不能更改 那么e的类型为const int &return 0;
}
- auto关键字的使用限制
- 1:函数形参类型无法推导(C++14标准支持)。
- 2:类的成员变量无法推导。
void foo(auto v){}
2、类型计算
2.1 类型计算分类
c语言:sizeof-计算类型的大小
C++语言:typeid-可以获取类型的信息字符串
C++11:decltype-获取参数表达式的类型
注意事项:类型推导和类型计算都是由编译器确定,并不是运行期确定
/*类型计算*/
int main(){const int a = 10;auto b = a; // 类型推导cout << "b的类型" << typeid(b).name() << endl; cout << "&b:" << &b << "&a:" << &a << endl; b++;// 允许更改,所以推导出来的类型是intdecltype(a)c = 100;// 类型计算 初始值可以设置和a不一样cout << "c的类型" << typeid(c).name() << endl;cout << "&c:" << &c << "&a:" << &a << endl;// // 地址不相同相同// c++; // 出错,说明不能更改 那么c的类型为const int return 0;
}
- 类型推导和类型计算的比较
1:类型计算比类型推导在类型的确定上更加精准
2:类型计算比类型推导在初值的确定上更加灵活
2.2 类型计算的四种规则
- 1:如果给decltype传递的为标识符表达式,decltype取该标识符的类型作为最终计算出的类型
int main(){int a = 10;// 如果给decltype传递的为标识符表达式,decltype取该标识符的类型作为最终计算出的类型decltype(a)b = a;// 类型计算cout << "b的类型" << typeid(b).name() << endl;cout << "&b:" << &b << "&a:" << &a << endl;// // 地址不相同b++; // 能更改说明b的类型为 int return 0;
}
- 2:如果给decltype传递的为函数表达式,decltype取该函数的返回值类型作为最终计算出的类型
float foo(){cout << "函数被调用" << endl;return 3.14;
}
int main(){int a = 10;// 如果给decltype传递的为函数表达式,decltype取该函数的返回值类型作为最终计算出的类型decltype(foo())b = a;// 并不会去实际调用foo()函数,类型计算是编译器确定的,不是运行时cout << "b的类型" << typeid(b).name() << endl;cout << "&b:" << &b << "&a:" << &a << endl;// // 地址不相同b++; // 能更改说明b的类型为 float return 0;
}
- 3:如果给decltype传递的为其他表达式,并且表达式的结果为左值,则取该左值引用的类型作为最终计算出的类型
int main(){int a = 10;// 如果给decltype传递的为其他表达式,并且表达式的结果为左值,则取该左值引用的类型作为最终计算出的类型decltype(++a)b = a;cout << "b的类型" << typeid(b).name() << endl;cout << "&b:" << &b << "&a:" << &a << endl;// // 地址相同b++; //允许更改 说明b的类型为int &return 0;
}
- 4:如果给decltype传递的为其他表达式,并且表达式的结果为右值,则取该右值本身的类型作为最终计算出的类型
int main(){int a = 10;// 如果给decltype传递的为其他表达式,并且表达式的结果为右值,则取该右值本身的类型作为最终计算出的类型decltype(a++)b = a;cout << "b的类型" << typeid(b).name() << endl;cout << "&b:" << &b << "&a:" << &a << endl;// // 地址不相同b++; //允许更改 说明b的类型为int return 0;
}
2.3 返回值类型计算
- 返回值类型后置
auto foo(int x, double y)->decltype(x+y){// 返回值类型后置,通过decltype计算得出return x + y;
}
int main(){auto f = foo(1, 3.1);// 类型推导cout << typeid(f).name() << endl; // double类型return 0;
}
3、列表初始化
基本类型,类类型,结构/联合/枚举类型等等的单个对象或对象数组,都可以采用形式完全统一的列表初始化语法
进行对象的初始化
- 书写形式:类型 对象 {初值表};
-int a{123);
-new double {1.23);
-string c{“123”};
-struct Student {d,“张飞”,20,{1997,10,10}};
-float e[]{1.1,2.2,3.3};
struct BD{int m_year;int m_month;int m_day;
};
struct myStudent{string m_name;int m_age;BD m_body;
};
class Human{
public:Human(int age = 0, const char* name = "无名") :m_age(age), m_name(name){}int m_age;string m_name;
};
int main(){int a = { 123 }; // int a=123cout << "a=" << a << endl;double* pa = new double{ 3.14 };// double *pa = new double(3.14);double *pb{ new double{ 3.14 } };cout << "*pa=" << *pa << " *pb=" << *pb << endl;int b[]{1, 2, 3};//int b[] = { 1, 2, 3 };for (int i = 0; i < 3; i++){ cout << b[i] << ' '; }int *parr{ new int[3]{4, 5, 6} };for (int i = 0; i < 3; i++){ cout << parr[i] << ' '; }delete[]parr;myStudent s{ "zs", 22, { 1997, 5, 7 } }; // myStudent s = { "zs", 22, { 1997, 5, 7 } };cout << s.m_name << s.m_age << s.m_body.m_year << s.m_body.m_month << s.m_body.m_day << endl;Human h{ 20, "赵云" };//Human h (20, "赵云" )cout << h.m_age << h.m_name << endl;return 0;
}
- 小括号操作符函数
class AA{
public:int operator()(int x, int y){return x + y;}
};
int main(){AA a;cout << a(100, 200) << endl;;// a.operator()(100,200)
}
4、Lambda表达式
4.1 前置知识
在C++中函数的作用域中可以有类型,也可以有表达式
- 针对于函数内部定义了类型,编译器先编译函数内部的类型,然后在编译函数体本身的代码
int a;
void foo(int b){int c;class A{public:void bar(int d){a = 0;// 能访问// b = 0; // 不能访问// c = 0; // 不能访问d = 0;// 能访问}};
}
4.2 Lambda表达式
- 语法规则:
[捕获表](参数表)选项->返回类型
{函数体
}
- 使用
int main(void){int a = 10, b = 20;auto c =[](int x, int y)->int{return x > y ? x : y; };// 编译器 (1) 生成一个类 (2) 类内定义一个小括号操作符函数 (3) 返回这个类的匿名对象cout<<c(a,b)<<endl;return 0;
}
- 本质
lambda表达式本质其实是一个类并且最终返回值为这个类的对象,因此对lambda表达式的调用就是该对象的函数操作符的调用
解释:编译器在编译到Lambda表达式时,编译器会生成一个类Z4XXX的类,类中定义一个小括号操作函数,函数体里面填充Lambda的函数体内容,函数的返回值类型为Lambda中定义的返回类型。
其中
- 可以没有返回值类型,将根据return推断
int main(void){int a = 10, b = 20;auto c = [](int x, int y){return x + y; };// 当没有返回值类型时,中间的箭头可以省略cout << c(a, b) << endl;return 0;
}
- 如果连return也没有,则返回值为void
int main(void){int a = 10, b = 20;[](int x, int y){cout<< x + y<<endl; }(a,b);// 如果连return也没有,则返回值为voidreturn 0;
}
- 参数为void可以省略不写的
int main(void){int a = 10, b = 20;[]{cout << "12345" << endl; }();// 如果没有形参,返回值类型也为void那么小括号和中间的剪头都可以省略return 0;
}
4.3 捕获表
- []-不捕获任何外部变量
- [variable]-捕获外部变量的值
- [&variable]-按引用捕获,外部变量的别名
- [this]-捕获this指针,访问外部对象的成员
int a = 10;
class Y{
public:Y(int m_e) :e(m_e){};void foo(int c = 30){cout << "---" << endl;// []-不捕获外部变量的值[](int d = 40){cout << "a=" << a << endl;cout << "b=" << b << endl;// cout << "c=" << c << endl; // 错误cout << "d=" << d << endl;// cout << "e=" << e << endl; // 错误}();cout << "--------------[c]--------" << endl;// [variable]-捕获外部变量只读[c](int d = 0){cout << "c=" << c << endl; }();cout << "--------[&c]---------" << endl;// [&variable]-按引用捕获,外部变量的别名[&c]{c++; cout << "c=" << c << endl; }();cout << c << endl;// [this]-捕获this指针,访问外部对象的成员cout << "--------[this]---------" << endl;[this]{cout << "e=" << e << endl; }();}
private:static int b;int e;
};
int Y::b = 10;int main(){Y y(4);y.foo();return 0;
}
- [=]-按值捕获所有的外部变量,也包括this
- [&]-按引用捕获所有的外部变量,也包括his
- [=,&variable]-按值捕获所有的外部变量包括this,但是指定的外部变量按引用捕获。
- [&,=variable]-按引用捕获所有的外部变量,也包括this,但是指定的外部变量按值捕获。
5、右值引用
5.1 概念
左值引用是别名,右值引用就是真名
左值:可以“取”地址的值就是左值,左值通常具名
右值:不可“取”地址的值就是右值,右值通常匿名
左值细分为非常左值和常左值
- 非常左值:有名字、可以取地址、没有常属性
- 常左值:有名字、可以取地址、有常属性
右值细分为纯右值和将亡值
- 纯右值:有一块无名内存,里面存放了基本类型的数据
- 将亡值:有一块无名内存,里面存放了类类型的数据
5.2 左值引用和右值引用
- 左值引用只能引用左值,不能引用右值
- 右值引用只能引用右值,不能引用左值
- 常左值引用,既能引用左值,也能引用右值
- 常右值引用,完全可以被常左值引用替代
/*左值引用和右值引用的差别*/
int main(){// 左值引用只能引用左值,不能引用右值int a=1, c=2;int & ra = a;// int & rb = a + c; // 错误 左值引用不能引用右值int &&rb = a + c;// 使用右值引用// int &&rc = a;// 错误 右值引用只能引用右值,不能引用左值cout << rb << endl;const int & _l = a; // 引用左值const int & _r = a + c; // 引用右值// 常左值引用会丧失修改目标的权限// _l = 9;// 错误// 右值引用不会丧失修改目标的权限rb = 90;cout << rb << endl;
}
6、移动语义
- 方法:
资源的转移 代替 资源的重建 - 作用:
保证功能正确的情况下,做到性能提升
//深拷贝构造函数 资源的重建String(const String & that) :m_psz(new char[strlen(that.m_psz) + 1]){cout << "深拷贝构造 资源的重建" << endl;strcpy(m_psz, that.m_psz); // 没有复制地址,复制了数据(深拷贝)
}
// 深拷贝构造函数 资源的转移
String(String && that):m_psz(that.m_psz){that.m_psz = NULL;cout << "资源转移" << endl;
}
// 深拷贝赋值函数 资源的重建String& operator=(/*String* this*/const String & that){cout<<"深拷贝赋值函数 资源的重建"<<endl;if (this == &that){}// 防止出现用户自己给自己赋值else{delete[] this->m_psz;// 编译器会先定义一个m_psz,并初始>化为空串,所以需要先释放内存this->m_psz = new char[strlen(that.m_psz) + 1];// 申请新资源strcpy(m_psz, that.m_psz); // 拷贝新内容}return *this;// 返回自引用
}// 拷贝赋值函数 资源的转移
String& operator=(String&& that){cout<<"拷贝赋值函数 资源的转移"<<endl;delete this->m_psz;this->m_psz=that.m_psz;that.m_psz=NULL;return *this;
}
//在linux下 命令行输入:g++ abc.cpp -std=c++11 -fno-elide-constructors
int main(){ String s1=String("hello"); // 深拷贝构造函数 资源的转移String s2=s1;// 拷贝构造函数 资源的重建s2=s1;// 拷贝赋值函数 资源的重建s2=String("hello");// 拷贝赋值函数 资源的转移
}
相关文章:

10 C++11
10 C11 1、类型推导1.1 auto关键字1.2 auto类型推断本质 2、类型计算2.1 类型计算分类2.2 类型计算的四种规则2.3 返回值类型计算 3、列表初始化4、Lambda表达式4.1 前置知识4.2 Lambda表达式4.3 捕获表 5、右值引用5.1 概念5.2 左值引用和右值引用 6、移动语义 1、类型推导 1…...
java的封装
为什么要封装?在java的面向对象的思想中,封装是指将类的实现细节包装,隐藏起来的方法。封装可以防止本类的代码和数据被外部定义的代码随机访问。 如何进行封装? 在定义一个类时,将类中的属性私有化,即使…...

为什么选择海外服务器?
如何选择跨境电商服务器:详细指南 选择合适的服务器是跨境电商企业成功的基础。服务器的性能和稳定性直接影响着网站的访问速度、用户体验和安全性,进而影响着企业的销量和利润。那么,跨境电商企业该如何选择服务器呢? 1. 确定目…...

k8s+springcloud+nacos部署配置
1 k8s 部署nacos-2.1.2配置k8s-nacos-statefulSet.yaml文件 apiVersion: v1 kind: Service metadata:name: nacos-headlessnamespace: rz-dtlabels:app: nacosannotations:service.alpha.kubernetes.io/tolerate-unready-endpoints: "true" spec:# 3个端口打开&…...

梯度提升决策树(GBDT)
GBDT(Gradient Boosting Decision Tree),全名叫梯度提升决策树,是一种迭代的决策树算法,又叫 MART(Multiple Additive Regression Tree),它通过构造一组弱的学习器(树&am…...
数据结构之B树的原理与业务场景
B树是一种自平衡的树形数据结构,它能够保持数据有序,并且可以高效地进行查找、顺序访问、插入和删除操作。B树的设计是为了优化磁盘I/O操作,因为它可以减少磁盘访问次数,这在数据库和文件系统中非常有用。 1. B树的原理 节点的出…...
【Android面试八股文】你能说一说线程池管理线程的原理吗?
面试官(Interviewer): 欢迎参加面试,今天我们会讨论一些关于 Java 线程池管理的问题。你能给我解释一下 ThreadPoolExecutor 是如何管理线程的吗? 候选人(Candidate): 当然可以,ThreadPoolExecutor 是 Java 中用于创建和管理线程池的核心类。它通过一组核心参数来控制线…...

springer 在线投稿编译踩坑
springer投稿,在线编译踩坑总结 注意: 有的期刊需要双栏,而预定义的模板中可能为单栏,需要增加iicol选项。 例如: \documentclass[sn-mathphys-num]{sn-jnl}% —>\documentclass[sn-mathphys-num, iicol]{sn-jnl}…...
固态硬盘的指标
固态硬盘的指标主要包括以下几个方面: 接口类型:这是固态硬盘与外部设备连接的方式,常见的接口类型有SATA、PCIe和NVMe等。不同的接口类型决定了固态硬盘的传输速度和性能。例如,PCIe接口的固态硬盘通常比SATA接口的固态硬盘具有…...
mysql 分组后每个取最新的一条记录
在MySQL中,若要从一个分组中获取每组的最新一条记录(通常基于时间戳或其他递增的列),可以使用子查询或者窗口函数(如果MySQL版本支持)。 以下是两种不同的实现方法: 方法1: 使用子查询和LIMIT…...

Java语法和基本结构介绍
Java语法和基本结构是Java编程的基础,它决定了Java代码的书写方式和程序的结构。以下是Java语法和基本结构的一些关键点: 1.标识符和关键字:Java中的标识符是用来标识变量、函数、类或其他用户自定义元素的名称。关键字是预留的标识符&#x…...
TDengine 3.3.0.0 引入图形化管理工具、复合主键等 13 项关键更新
在涛思数据研发团队的努力下,TDengine 3.3.0.0 版本终于和大家见面了。这一版本中,我们引入了多项革新功能和性能优化,力求在为用户提供极致体验的同时,不断推动技术的前沿。 此次更新不仅针对开源社区版本,进行了一系…...
C++基础之红黑树
二叉搜索树 二叉搜索树(Binary Search Tree,BST)是一种二叉树,具有以下性质: 左子树节点值小于根节点值:对于树中的每个节点 x,其左子树中所有节点的值都小于 x 的值。右子树节点值大于根节点值…...
ClickHouse数据库对比、适用场景与入门指南
本文全面对比了ClickHouse与其他数据库(如StarRocks、HBase、MySQL、Hive、Elasticsearch等)的性能、功能、适用场景,并提供了ClickHouse的教学入门指南,旨在帮助读者选择合适的数据库产品并快速掌握ClickHouse的使用。 文章目录 …...

举例说明 如何通过SparkUI和日志定位任务莫名失败?
有一个Task OOM: 通过概览信息,发现Stage 10的Task 36失败了4次导致Job失败。概览信息中显示最后一次失败的退出代码(exit code)是143,意味着发生了内存溢出(OOM,即Out of Memory)。…...
Vue前端通过Axios的post方式传输数据,后端为什么一直接收的值是null?
沃靠!这个细节太细了,搞了我两个多小时才找到这个bug。 一、 首先官方文档给我的post请求的例子是这样的: axios.post(/user, {firstName: Fred,lastName: Flintstone}).then(function (response) {console.log(response);}).catch(function (error) {console.log(error);})…...

外链建设如何进行?
理解dofollow和nofollow链接,所谓dofollow链接,就是可以传递权重到你的网站的链接,这种链接对你的网站排名非常有帮助,这种链接可以推动你的网站在搜索结果中的位置向上爬,但一个网站全是这种有用的链接,反…...
深入理解Java正则表达式及其应用
正则表达式是一种强大的文本匹配和处理工具,可以在字符串中查找、替换、提取符合特定模式的内容。Java作为一种广泛应用的编程语言,提供了丰富的正则表达式支持。本文将深入探讨Java正则表达式的基本概念、语法以及常见应用场景,帮助读者全面…...

Gstreamer学习3----灌数据给管线之appsrc
参考资料 Basic tutorial 8: Short-cutting the pipeline gstreamer向appsrc发送帧画面的代码_gst appsrc可变帧率-CSDN博客 在官网教程Basic tutorial 8: Short-cutting the pipeline 里面,讲了一个例子,push音频数据给管线,视频的例子更…...

【深度学习量化交易1】一个金融小白尝试量化交易的设想、畅享和遐想
关注我的朋友们可能知道,我经常在信号处理的领域出没,时不时会发一些信号处理、深度学习科普向的文章。 不过算法研究久了,总想做一些更有趣的事情。 比如用深度学习算法赚大钱。。毕竟有什么事情能比暴富更有意思呢。 一、神经网络与彩票…...

使用VSCode开发Django指南
使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...
《Playwright:微软的自动化测试工具详解》
Playwright 简介:声明内容来自网络,将内容拼接整理出来的文档 Playwright 是微软开发的自动化测试工具,支持 Chrome、Firefox、Safari 等主流浏览器,提供多语言 API(Python、JavaScript、Java、.NET)。它的特点包括&a…...

剑指offer20_链表中环的入口节点
链表中环的入口节点 给定一个链表,若其中包含环,则输出环的入口节点。 若其中不包含环,则输出null。 数据范围 节点 val 值取值范围 [ 1 , 1000 ] [1,1000] [1,1000]。 节点 val 值各不相同。 链表长度 [ 0 , 500 ] [0,500] [0,500]。 …...
基础测试工具使用经验
背景 vtune,perf, nsight system等基础测试工具,都是用过的,但是没有记录,都逐渐忘了。所以写这篇博客总结记录一下,只要以后发现新的用法,就记得来编辑补充一下 perf 比较基础的用法: 先改这…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...

select、poll、epoll 与 Reactor 模式
在高并发网络编程领域,高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表,以及基于它们实现的 Reactor 模式,为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。 一、I…...

【VLNs篇】07:NavRL—在动态环境中学习安全飞行
项目内容论文标题NavRL: 在动态环境中学习安全飞行 (NavRL: Learning Safe Flight in Dynamic Environments)核心问题解决无人机在包含静态和动态障碍物的复杂环境中进行安全、高效自主导航的挑战,克服传统方法和现有强化学习方法的局限性。核心算法基于近端策略优化…...

三分算法与DeepSeek辅助证明是单峰函数
前置 单峰函数有唯一的最大值,最大值左侧的数值严格单调递增,最大值右侧的数值严格单调递减。 单谷函数有唯一的最小值,最小值左侧的数值严格单调递减,最小值右侧的数值严格单调递增。 三分的本质 三分和二分一样都是通过不断缩…...

【Linux】Linux安装并配置RabbitMQ
目录 1. 安装 Erlang 2. 安装 RabbitMQ 2.1.添加 RabbitMQ 仓库 2.2.安装 RabbitMQ 3.配置 3.1.启动和管理服务 4. 访问管理界面 5.安装问题 6.修改密码 7.修改端口 7.1.找到文件 7.2.修改文件 1. 安装 Erlang 由于 RabbitMQ 是用 Erlang 编写的,需要先安…...
文件上传漏洞防御全攻略
要全面防范文件上传漏洞,需构建多层防御体系,结合技术验证、存储隔离与权限控制: 🔒 一、基础防护层 前端校验(仅辅助) 通过JavaScript限制文件后缀名(白名单)和大小,提…...