摘录

constexptr

C++14尝鲜:constexpr函数(编译期函数)

总结来说,就是在c++11之前,要实现编译期数值计算需要繁琐的模板元编程。在c++11 中,可以是函数,在一句ruturn 语句中进行求值,函数中既不能有变量也不能有分之判断语句,限制较多。在C++17之后,则取消了大部分限制,比如可以有变量,可以有分支判断语句,但不能有goto,asm,try等语句。具体可以参考cppreference

准确的说,constexpr函数是一种在编译期和运行期都能被调用并执行的函数。出于constexpr函数的这个特点,在C++11之后进行数值计算时,无论在编译期还是运行期我们都可以统一用一套代码来实现。编译期和运行期在数值计算这点上得到了部分统一。

什么是rvalues,lvalues,xvalues,glvalues和prvalues?

  • lvalue(所谓的,历史上,因为左值可以出现在赋值表达式的左侧)表示一函数或一个对象。 [示例:如果E是指针类型的表达式,则*E 是一个左值表达式,引用要指向的对象或函数E 。另一个例子,调用返回类型为左值引用的函数的结果是左值。
  • xvalue(一个“到期”值)也指一个对象,通常接近其寿命的末尾(使得其资源可以被移动,例如)。xvalue是涉及rvalue引用的某些表达式的结果。 [示例:调用返回类型为右值引用的函数的结果是xvalue。]
  • glvalue (“广义”左值)是左值 或x值。
  • rvalue(所谓的,历史上,因为右值可以出现在赋值表达式的右手侧)是x值,临时对象或子对象,或它们的不与一个对象相关联的值。
  • prvalue(“纯”右值)是一个rvalue,这不是一个x值。 [示例:调用返回类型不是引用的函数的结果是prvalue]

lvalue 左值

当且仅当E指的是已经具有使其可在E外部访问的标识(地址,名称或别名)的实体时,表达式E属于左值类别。

#include <iostream>

int i=7;

const int& f(){
return i;
} int main()
{
std::cout<<&"www"<<std::endl; // This address ...
std::cout<<&"www"<<std::endl; // ... and this address are the same.
"www"; // The expression "www" in this row is an **lvalue expression**, because it refers to the same entity ...
"www"; // ... as the entity the expression "www" in this row refers to. i; // The expression i in this row is an lvalue expression, because it refers to the same entity ...
i; // ... as the entity the expression i in this row refers to. int* p_i=new int(7);
*p_i; // The expression *p_i in this row is an lvalue expression, because it refers to the same entity ...
*p_i; // ... as the entity the expression *p_i in this row refers to. const int& r_I=7;
r_I; // The expression r_I in this row is an lvalue expression, because it refers to the same entity ...
r_I; // ... as the entity the expression r_I in this row refers to. f(); // The expression f() in this row is an lvalue expression, because it refers to the same entity ...
i; // ... as the entity the expression f() in this row refers to. return 0;
}

xvalues 亡值

若表达式E属于xvalue类别,当且仅当它满足:

-无论是隐式调用还是显示调用,调用一个函数的结果返回一个对要返回的对象类型的引用的rvalue


int&& f(){
return 3;
} int main()
{
f(); // 表达式 f() 是xvalue ,因为 f() 的返回类型是一个对象类型的引用的 rvalue. return 0;
}

-或者,右值(rvalue)引用对象类型的cast

int main()
{
static_cast<int&&>(7); // 表达式static_cast<int&&>(7) , 其是右值引用对象类型的 cast.
std::move(7); // std::move(7) is equivalent to static_cast<int&&>(7). return 0;
}

-类成员访问表达式指定 非引用类型的非静态数据成员, 其中对象表达式是 xvalue

struct As
{
int i;
}; As&& f(){
return As();
} int main()
{
f().i; // f().i 是 xvalue, 因为 As::i 是 non-static data member of non-reference type, and the subexpression f() belongs to the xvlaue category. return 0;
}
  • 一种pointer-to-member表达式, 其中第一个操作数是 xvalue, 第二个操作数是指向数据成员的指针。

    请注意,上述规则的效果是对命名rvalue引用被视为lvalues,对对象的未命名rvalue引用被视为xvalues; 对函数的右值引用被视为左值,无论是否命名。
#include <functional>

struct As
{
int i;
}; As&& f(){
return As();
} int main()
{
f(); // The expression f() belongs to the xvalue category, because it refers to an unnamed rvalue reference to object.
As&& rr_a=As();
rr_a; // rr_a 是 lvalue category,因为它指的是一个命名的 rvalue 对象引用
std::ref(f); // n std::ref(f) 表达式是 lvalue , 因为它指的是一个命名的 rvalue 对象引用 return 0;
}

prvalue

当且仅当E既不属于左值也不属于xvalue类别时,表达式E属于prvalue类别。

struct As
{
void f(){
this; // this 是 **prvalue** 表达式。 注意:this 不是一个变量
}
}; As f(){
return As();
} int main()
{
f(); // f() 表达式属于 prvalue category, 其既不属于 lvalue 也不属于 xvalue 类 return 0;
}

rvalue

当且仅当E属于xvalue类别或属于prvalue类别时,表达式E属于rvalue类别。

请注意,此定义表示当且仅当E指的是没有任何标识使其可在E之外访问的实体时,表达式E属于右值类别。

glvalues

当且仅当E属于lvalue类别或xvalue类别时,表达式E属于glvalue类别。

一个实用的规则

Scott Meyers 已出版的经验的一个非常有用的规则,从左值右值的区别。

如果可以获取表达式的地址,则表达式为左值。

如果表达式的类型是左值引用(例如,T&或const T&等),则该表达式是左值。

否则,表达式是右值。从概念上(通常也是实际上),rvalues对应于临时对象,例如从函数返回的或通过隐式类型转换创建的临时对象。大多数文字值(例如,10和5.3)也是右值。

C++ 新特性 笔记的更多相关文章

  1. html新特性笔记

    HTML5知识总结 l  文档类型声明:<!DOCTYPE HTML> l  新绘制元素: Canvas:标签定义图形,比如图表和其他图像.该标签基于 JavaScript 的绘图 API ...

  2. C++ 新特性 笔记 2 右值引用

    C ++ Rvalue引用说明 以下内容,主要是上述链接的摘要 介绍 Rvalue引用是C ++的一个特性,它是随C ++ 11标准添加的.使右值参考有点难以理解的是,当你第一次看到它们时,不清楚它们 ...

  3. ES 6 新特性笔记

    let 与 var 的区别 功能 let var 块级作用域 ️ 变量提升 ️ 重复声明(相同作用域内) ️ var 没有块级作用域的解决方法 使用函数替代块级作用域,以保证变量的正常使用,如: .. ...

  4. CSS/CSS3语法新特性笔记

    CSS层叠样式表 三大特性 层叠性:相同的样式会覆盖 继承性:属性可向下继承 优先级:范围越小权重越高 选择器 基础选择器 标签选择器 1 body { 2 color:#fff; 3 } 类选择器 ...

  5. java8新特性笔记

    1.forEach(),遍历数据结构中的元素,括号内可以带一个闭包的方法 2.双冒号用法:forEach(this::doSchedule),如果运行环境是闭包,java允许使用双冒号的写法来直接调用 ...

  6. MySQL5.7新特性笔记

    001.用户验证方式上的变化 MySQL5.7已经不要支持mysql_old_password验证插件,也不再支持old_password函数,也就说old_passwords也就不能被设置成1了.

  7. ES6 新特性(笔记)

    1.定义变量     let         a).块作用域 , 不同于var的函数作用域         b).不可以重复定义同一个变量名           注:              {} ...

  8. C#新特性记录

    C#6.0新特性笔记 Getter专属赋值 可以在构造函数中,给只有get的属性赋初始值. class Point { public int x { get; } public Point() { x ...

  9. C++ 11学习和掌握 ——《深入理解C++ 11:C++11新特性解析和应用》读书笔记(一)

    因为偶然的机会,在图书馆看到<深入理解C++ 11:C++11新特性解析和应用>这本书,大致扫下,受益匪浅,就果断借出来,对于其中的部分内容进行详读并亲自编程测试相关代码,也就有了整理写出 ...

随机推荐

  1. 树莓派-CentOS-Minimal arm版的设置

    将镜像用 balenaEtcher 写入到树莓派SD卡并启动后,需要对其进行一些设置才能正常使用. 1. 用户名 root 密码 centos 2. 扩展 rootfs 到最大可用空间:cat REA ...

  2. 代码托管至Github

    昨天突然之间觉得作为一个iOS程序员,没有在github上提交过自己的代码真是一大遗憾,不管是自己写的优秀的代码还是刚开始学习,用来学习练手的项目.然后我就很想要学习怎么往github上提交代码,很不 ...

  3. flask(1.1)装饰器装饰多个视图函数出现的问题

    目录 1.装饰器装饰多个视图函数出现的问题 2.使用装饰器修复技术解决该问题 1.装饰器装饰多个视图函数出现的问题 代码实例: from flask import Flask, request, re ...

  4. Lua易忘点

    仅针对自己 __index的理解 __index是:当我们访问一个表中的元素不存在时,则会触发去寻找__index元方法,如果不存在,则返回nil,如果存在,则返回结果 Window = {} Win ...

  5. 快速搭建WordPress博客

    博主在看了朋友的博客后 决定也搭建一个wordPress 博客 思路 1.购买服务器 2.Cenots环境配置 3.安装wordpress 工具 推荐使用 Xshell 6,当然也可以用其他 服务器推 ...

  6. 配置glance使用ceph作为后端存储

    在ceph监视器上执行 1.创建pool池 为glance服务创建pool池(因为我只有一个OSD节点,所以要将副本数设置为1) ceph osd pool create glance-images  ...

  7. CentOS学习之NTP服务配置详解

    详解centos7下ntp服务配置 一.ntp服务是什么 1.定义 NTP是网络时间协议(Network Time Protocol),它是用来同步网络中各个计算机的时间的协议. 2.发展 首次记载在 ...

  8. Linux多线程编程 - sleep 和 pthread_cond_timedwait

    #include <stdio.h> #include <stdlib.h> int flag = 1; void * thr_fn(void * arg) {   while ...

  9. 骨牌摆放方案数n*m(状压DP)

    题意:https://www.nitacm.com/problem_show.php?pid=1378 如题. 思路: 从第一行for到最后一行,枚举每一行的所有状态,进行转移,注意答案是dp[最后一 ...

  10. X86逆向15:OD脚本的编写技巧

    本章节我们将学习OD脚本的使用与编写技巧,脚本有啥用呢?脚本的用处非常的大,比如我们要对按钮事件进行批量下断点,此时使用自动化脚本将大大减小我们的工作量,再比如有些比较简单的压缩壳需要脱壳,此时我们也 ...