C++中关于左值和右值的讨论

左值性(lvalueness)在C/C++中是表达式的一个重要属性。只有通过一个左值表达式才能来引用及更改一个对象(object)的值。(某些情况下,右值表达式也能引用(refer)到某一个对象,并且可能间接修改该对象的值,后述)。

何谓对象?如果没有明确说明,这里说的对象,和狭义的类/对象(class/object)相比,更为广泛。在C/C++中,所谓的对象指的是执行环境中一块存储区域(a region of storage),该存储区域中的内容则代表(represent)了该对象的值(value)。注意到我们这里所说的"代表",对于一个对象,如果我们需要取出(fetch)它的值,那么我们需要通过一定的类型(type)来引用它。使用不同的类型,对同一对象中的内容的解释会导致可能得到不同的值,或者产生某些未定义的行为。

在介绍左值之前,我们还需要引入一个概念: 变量(variable)。经常有人会把变量与对象二者混淆。什么叫变量?所谓变量是一种声明,通过声明,我们把一个名字(name)与一个对象对应起来,当我们使用该名字时,就表示了我们对该对象进行某种操作。但是并不是每个对象都有名字,也并不意味着有对应的变量。比如临时对象(temporary object)就没有一个名字与之关联(不要误称为临时变量,这是不正确的说法)。

1 C中的左值

1.1按照C的定义,左值是一个引用到对象的表达式,通过左值我们可以取出该对象的值。通过可修改的左值表达式(modifiable lvalue)我们还可以修改该对象的值。(需要说明的是,在C++中,左值还可以引用到函数,即表达式f如果引用的是函数类型,那么在C中它既不是左值也不是右值;而在C++中则是左值)。因为左值引用到某一对象,因此我们使用&对左值表达式(也只能对左值表达式和函数)取址运算时,可以获得该对象的地址(有两种左值表达式不能取址,一是具有位域( bit-field )类型,因为实现中最小寻址单位是 byte;另一个是具有register指定符,使用register修饰的变量编译器可能会优化到寄存器中)。

Ex1.1
char a[10];  // a is an lvalue representing an array of 10 ints.
char (* p)[10]=&a; // &a is the address of the array a.
const char* p="hello world"; //"hello world" is an lvalue of type char[12]
//in C,  type const char[12] in C++.
char (*p)[12]=&"hello world";

struct S{ int a:2; int b: 8; };
struct S  t;
int* p=&t.a; //error. t.a is an lvalue of bitfield.

register int i;
int * p=&i; //error. i is an lvalue of register type.
int a, b;
int * p=& (a+b); //error. a+b is not an lvalue.

1.2假设expr1是一个指向某对象类型或未完整类型(incomplete type,即该类型的布局和大小未知)的指针,那么我们可以断言*expr1一定是个左值表达式,因为按照*运算符的定义,*expr1表示引用到expr1所指向的对象。如果expr1是个简单的名字,该名字代表一个变量。

同样的,该表达式也是个左值,因为他代表的是该变量对应的对象。对于下标运算符,我们一样可以做出同样的结论,因为expr1[expr2]总是恒等于*( ( expr1 )+ expr2 ),那么p->member,同样也是一个左值表达式。然而对于expr1.expr2,则我们不能断定就是个左值表达式。因为expr1可能不是左值。

需要特别说明的是,左值性只是表达式的静态属性,当我们说一个表达式是左值的时候,并不意味着它一定引用到某一个有效存在的对象。int *p; *p是左值表达式,然而这里对*p所引用的对象进行读写的结果将可能是未定义的行为。

Ex1.2
extern struct A a;
struct A* p2= &a;

a是个左值表达式,因而可以进行&运算,然而此时stru A仍然没有完整。

//In C++
extern class A a;
A & r=a;// OK. Refers to a, though a  with an incomplete type.

1.3可修改的左值

在语义上需要修改左值对应的对象的表达式中,左值必须是一个可修改的左值。比如赋值(包括复合赋值)表达式中的左操作数,必须是一个可修改的左值表达式;自增/减运算符的操作数等。

Ex1.3
const int a[2], i; //NOTE: a unintialized. legal in C, illegal in C++.
i++; //error, i is an lvalue of type const int.
a[0]--;//error, a[0] is an lvalue of const int.

时间: 2024-10-14 21:03:09

C++中关于左值和右值的讨论的相关文章

c++中的左值与右值

转载自 http://www.cnblogs.com/catch/p/3500678.html 左值 (lvalue)和右值 (rvalue) 是 c/c++ 中一个比较晦涩基础的概念,有的人可能甚至没有听过,但这个概念到了 c++11 后却变得十分重要,它们是理解 move, forward 等新语义的基础. 左值右值的定义 左值与右值这两概念是从 c 中传承而来的,在 c 中,左值指的是既能够出现在等号左边也能出现在等号右边的变量(或表达式),右值指的则是只能出现在等号右边的变量(或表达式)

c++-关于函数参数左值与右值?

问题描述 关于函数参数左值与右值? 这里:string s1(""hi"")s2s3;s2=std::move(string(""bye""));//正确:从一个右值移动数据s3=std::move(s1);书上说:在s2中传递给move的实参是string的构造函数的右值结果--string(""bye"").那可不可以直接传递""bye"" 呢

《C++语言入门经典》一2.8 左值与右值

2.8 左值与右值 C++中的每个语句.表达式的结果分为左值与右值两类.左值指的是内存中持续存储的数 据,而右值是指临时存储的结果. 在程序中,声明过的独立变量如: Int k; short p; char a; 它们都是左值.又如: Int a = 0; Int b = 2; Int c = 3;   a = c-b; b = a++; c = ++a; c--; c-b是一个存储表达式结果的临时数据,它的结果将被复制到a中,它是一个右值.a++自 增的过程实质上是一个临时变量执行了表达式,而

左值、右值与右值引用

在C语言中,我们常常会提起左值(lvalue).右值(rvalue)这样的称呼.而在编译程序时,编译器有时也会在报出的错误信息中会包含 左值.右值的说法.不过左值.右值通常不是通过一个严谨的定义而为人所知的,大多数时候左右值的定义与其判别方法是一体的.一个最为典型的判别方法就是, 在赋值表达式中,出现在等号左边的就是"左值",而在等号右边的,则称为"右值".比如: a = b + c; 在这个赋值表达式中,a就是一个左值,而b + c则是一个右值.这种识别左值.右值

c++ 11 移动语义、std::move 左值、右值、将亡值、纯右值、右值引用

为什么要用移动语义 先看看下面的代码 // rvalue_reference.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include <iostream> class HugeMem { public: HugeMem(int size) : sz(size) { pIntData = new int[sz]; } HugeMem(const HugeMem & h) : sz(h.sz) { pIntData =

数据库中的左连接和右连接的区别

今天,别人问我一个问题:数据库中的左连接和右连接有什么区别?如果有A,B两张表,A表有3条数据,B表有4条数据,通过左连接和右连接,查询出的数据条数最少是多少条?最多是多少条? 我被这个问题问住了,后来我去问了数据库开发人员,结果结果各种各样: a 最大12  最小0 b 最大12  最小未知 c 最大未知 最小为3 d 最大12   最小为3 e 不清楚 1.说明 (1)左连接:只要左边表中有记录,数据就能检索出来,而右边有 的记录必要在左边表中有的记录才能被检索出来 (2)右连接:右连接是只

C++中的左值和右值

在C/C++中,左值(lvalue)和右值(rvalue)是用于规定表达式(expression)的性质.C++中表达式要不然是左值,要不然是右值. 这两个概念在C语言中比较容易理解:左值能放在赋值语句的左边,右值不能.但是当来到C++时,二者的理解就比较复杂了(PS:有对象真是麻烦) 简单的归纳: 当一个对象被用作右值的时候,用的是对象的值(内容):当对象被用作左值的时候,用的是对象的身份即在内存中的地址. 左值是代表一个内存地址值,并且通过这个内存地址,就可以对内存进行读并且写(主要是能写)

SQLServer中字符串左对齐或右对齐显示的sql语句_MsSql

知识点: 函数 replicate 以下代码是实现如下功能: 复制代码 代码如下: declare @sql varchar(200), --需填充的字符串 @char varchar(4), --填充使用的字符 @len int --填充后的长度 select @sql='abc' select @char=' ' select @len=10 select (right(replicate(@char,@len)+@sql,@len)) 右对齐 ,@sql+replicate(@char,@

SQLServer中字符串左对齐或右对齐显示的sql语句

知识点: 函数 replicate 以下代码是实现如下功能: 复制代码 代码如下: declare @sql varchar(200), --需填充的字符串 @char varchar(4), --填充使用的字符 @len int --填充后的长度 select @sql='abc' select @char=' ' select @len=10 select (right(replicate(@char,@len)+@sql,@len)) 右对齐 ,@sql+replicate(@char,@