C++11 中的左值、右值和将亡值

C++98 中表达式值的类型只有左值和右值两种类型,可以取到地址的表达式就是左值,不是左值的值就是右值,而C++11中将表达式的值类型划分成了lvalue(左值)、rvalue(右值)、prvalue(纯右值)、xvalue(将亡值)、gvalue(泛左值) 5种。下文从基本概念并结合实际的例子区分C++11中的表达式值类型。

基本概念

表达式

  • 表达式的定义
    定义: 由运算符(operator)和运算对象(operand)构成的计算式(类似于数学上的算术表达式)。
    举例: 字面值(literal)和变量(variable)是最简单的表达式,函数的返回值也被认为是表达式。

值类别

  • 表达式是可求值的,对表达式求值将得到一个结果(result)。这个结果有两个属性:类型和值类别(value categories)。
  • 在c++11以后,表达式按值类别分,必然属于以下三者之一:左值(left value,lvalue),将亡值(expiring value,xvalue),纯右值(pure rvalue,pralue)。其中,左值和将亡值合称泛左值(generalized lvalue,glvalue),纯右值和将亡值合称右值(right value,rvalue)。见下图,


    642223-20160314175051818-1679937642.png
  • C++11中右值rvalue的概念包括两个部分:
    1. 将亡值(xvalue. expiring value),xvalue是C++11新增的概念,与右值引用相关的表达式,如: 将要被移动的对象、T&&函数返回值、std::move返回值和转换为 T&&的类型的转换函数的返回值等;
    2. 纯右值(pvalue, pure right value),如: 非引用返回的临时变量、运算表达式产生的临时变量、原始字面量和lambda表达式等属于pvalue;

注意:“左值”是表达式的结果的一种属性,但更为普遍地,通常用“左值”来指代左值表达式。所谓左值表达式,就是指求值结果的值类别为左值的表达式。在后文中,我们依然用左值指代左值表达式。对于纯右值和将亡值,亦然。

左值、纯右值和将亡值的描述

左值

  • 描述: 能够用&取地址的表达式是左值表达式。

一个区分左值和右值的便捷方法:看能不能对表达式取地址,若能,则为左值,若不能则为右值。所有的具名变量都是左值,而右值是不具名的。

  • 举例:
    • 函数名和变量名
    • 返回左值引用的函数调用
    • 前置自增/自减运算符连接的表达式++i/--i
    • 由赋值运算符或复合赋值运算符连接的表达式(a=b、a+=b、a%=b)
    • 解引用表达式*p
    • 字符串字面值"abc"

纯右值

  • 描述: 满足下列条件之一的:
    • 1)本身就是赤裸裸的、纯粹的字面值,如3、false;
    • 2)求值结果相当于字面值或是一个不具名的临时对象。
  • 举例:
    • 除字符串字面值以外的字面值
    • 返回非引用类型的函数调用
    • 后置自增/自减运算符连接的表达式i++/i--
    • 算术表达式(a+b、a&b、a<<b)
    • 逻辑表达式(a&&b、a||b、~a)
    • 比较表达式(a==b、a>=b、a<b)
    • 取地址表达式(&a)

将亡值

  • 描述: 在C++11之前的右值和C++11中的纯右值是等价的。C++11中的将亡值是随着右值引用的引入而新引入的。换言之,“将亡值”概念的产生,是由右值引用的产生而引起的,将亡值与右值引用息息相关。所谓的将亡值表达式,就是下列表达式:
    • 1)返回右值引用的函数的调用表达式
    • 2)转换为右值引用的转换函数的调用表达

在C++11中,用左值去初始化一个对象或为一个已有对象赋值时,会调用拷贝构造函数或拷贝赋值运算符来拷贝资源,而当用一个右值(包括纯右值和将亡值)来初始化或赋值时,会调用移动构造函数或移动赋值运算符来移动资源,从而避免拷贝,提高效率。当该右值完成初始化或赋值的任务时,它的资源已经移动给了被初始化者或被赋值者,同时该右值也将会马上被销毁(析构)。也就是说,当一个右值准备完成初始化或赋值任务时,它已经“将亡”了。这种右值常用来完成移动构造或移动赋值的特殊任务,扮演着“将亡”的角色,所以C++11给这类右值起了一个新的名字——将亡值。

  • 举例:std::move()、tsatic_cast<X&&>(x)(X是自定义的类,x是类对象,这两个函数常用来将左值强制转换成右值,从而使拷贝变成移动,提高效率。

C++11中值的类型总结

  • gvalue: has identity
  • lvalue: has identity and can not be moved from
  • rvalue: can be moved from
  • xvalue: has identity and can be moved from
  • prvalue: does not have identity and can be moved from

对比说明

情况1: ++i是左值,而i++是右值

  • ++i对i加1后再赋给i,最终的返回值就是i,所以,++i的结果是具名的,名字就是i;
  • 而对于i++而言,是先对i进行一次拷贝,将得到的副本作为返回结果,然后再对i加1,由于i++的结果是对i加1前i的一份拷贝,所以它是不具名的。假设自增前i的值是6,那么,++i得到的结果是7,这个7有个名字,就是i;而i++得到的结果是6,这个6是i加1前的一个副本,它没有名字,i不是它的名字,i的值此时也是7。
  • 可见,++i和i++都达到了使i加1的目的,但两个表达式的结果不同。

情况2: 解引用表达式*p是左值,取地址表达式&a是纯右值

  • &(p)一定是正确的,因为p得到的是p指向的实体,&(p)得到的就是这一实体的地址,正是p的值。由于&(p)的正确,所以*p是左值。
  • 而对&a而言,得到的是a的地址,相当于unsigned int型的字面值,所以是纯右值。

情况3: a+b、a&&b、a==b都是纯右值

  • a+b得到的是不具名的临时对象,而a&&b和a==b的结果非true即false,相当于字面值。

情况4: 字符串字面值是左值,而非字符串的字面量是纯右值

  • 不是所有的字面值都是纯右值,字符串字面值是唯一例外。
  • 早期C++将字符串字面值实现为char型数组,实实在在地为每个字符都分配了空间并且允许程序员对其进行操作,所以类似
    cout<<&("abc")<<endl;
    char *p_char="abc";//注意不是char *p_char=&("abc");
    

情况5: 具名的右值引用是左值,不具名的右值引用是右值。

  • 如:
    void foo(int&& t) {
        // t 被一个右值表达式初始化
        // 但是,t 本身是一个左值
    }
    

继续深入学习

C++11中的右值引用和移动语义

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容