`
Michaelmatrix
  • 浏览: 209496 次
  • 来自: 北京
文章分类
社区版块
存档分类
最新评论

自加 左值

 
阅读更多

1.前置运算符和后置运算符,左值和右值。

其实很久以来一直都没有怎么搞清楚左值和右值的区别,只知道左值可以放在等号的左边,也可以放在等号的右边,但是右值却只能放在等号的右边,然后形 成一个大概直观的印象,知道怎么样做才不出错而已。不过今天看看C++,却发现有了点新的体会。

对于表达式a--=5;这样一个表达式,明显是错误的,究其原因,是因为执行等号左边的自减表达式之后,显示取得a的值,然后才是进行自减操作,所 以最终的结果是一个右值,而且就是a的值(变化前的),于是对于赋值语句 b = a--,自然不会弄错了。

而对于--a=5;这样一个表达式,则明显就不同了。首先执行的是自减操作,然后返回的是a的值(变化后的),于是自然就可以将返回的a值再次赋值 了。也就是说前置的运算符是左值表达式,而后置的则是右值表达式。于是,我想到了很久前的一个想法:++a++ = 5;当然,行家肯定一看就知道是错误的,可是错误的地方需要改正啊。于是我想到了几种方法,就是通过加括号的方法来实现。

1)++(a++) = 5; 其实通过错误信息就可以看到原因了:error: non-lvalue in increment ,明显说明在自增运算符中的那个表达式不是左值,所以不能自增运算了。哦,对啊,a++是右值,于是的话再次进行++的操作就是错误的,因为++操作需要 的表达式是左值表达式。于是此处的方法行不通。

2)(++a)++ = 5;此处也不行,错误信息就是: error: non-lvalue in assignment,通过前面的表述就知道了,其实++a是左值表达式,那么后面的那个++操作就是一个错误了,赋值的时候,等号的左边不是左值表达 式,因为括号外的是一个后置的++运算符。于是,就得到了错误。

通过上面的这些折腾,对于++++a等等变态的表达式,相信也就有更深刻的体会了。

2.重载前置++和后置++的时候的不同

相信很多人都做过这种操作了,对于后置的++,需要多一个形参int来说明,但是从上面的分析,还需要注意的是两者的左值和右值的区别,重载的目 的,也不过是为了更加符合用户的习惯,于是,我们对于前置的++,自然要返回的是左值表达式,而对于后置的++,需要返回的则是左值表达式。当然,具体的 来说说吧。

前置++:如果自定义了一个类A(含有一个数据成员x),那么要重载它的前置++运算符,根据习惯,首先就是对它的数据成员++,然后再返回它的引 用,这样,才是左值表达式,于是下面的代码: A& A::operator++(){ ++x; return *this ; }这样的方法,自然很好的表达了前置的作用,先运算,在返回值。

后置++:如果同上面的例子,要定义后置++的话,那么需要注意的是它的右值性,此时是返回引用还是类类型呢?思考了很久,一般来说,返回的是引用 的,都是左值表达式,而如果此处返回的是引用,那么就可能在后面的代码中出现问题。

于是出现了下面的这样的代码:

A A::operator++(int )

{

return A(this->x++);

}

其实我觉得此句甚好,既表达了后置的意思,又成功的返回了值。后置的含义,也就是先返回之后再进行自增操作,于是,此处调用的是默认的拷贝构造函数 来实现返回一个临时的变量。

3.重载函数的参数问题

以前的时候被这个问题纠结的不行了,后来明白了一点,今天看看书,明白的更多了,现在才明白古人那句“书读百遍,其义自见”的至理名言了。也更加的 明白了读书的重要性。

对于运算符重载,我们常用的,既可以作为成员函数,也可以作为普通函数,这点都知道,而且,此时的参数不同,作为成员函数的话,因为在类中,所以有 一个隐藏的this指针,于是,双目运算符就只要写出来一个就可以,而单目运算符就不需要写出来。同样的是一个重载运算符,可以有两种表示的方法,如:A operator+(A&, int value); //此为普通的函数 A A::operator+(int value) ; //此为成员函数

对于上面的这两个典型的例子,很容易而直白的理解,在调用的时候,写下 a + 5; 其中a是A类型的变量,这样就OK了。但是深入思考下,重载的是+运算符,那么,双目的运算,那两个参数的顺序是,如何区分呢?很简单,前面的那个操作符 就是第一个操作符,而后面的那个就是第二个运算符。对于是成员的函数,那么就要注意this指针的作用,所以也就说明指代的对象的重要性了。为何我这么简 单的东西还要思考呢?因为被我们书上的一个变态程序搞懵了,如下:

#include <iostream>
using namespace std;
struct A{
int a;
A(int x){a = x;}
};

class B{
A x;
public:
A *operator->();
B(int v):x(v){}
};

A *B::operator->(){
return &x;
}

int main()
{
B b(5);
int i = b->a;
b->a = i+5;
i = (*b.operator->()).a;
}

各位可以看看,其实道理很简单,不过是重载了->,而且重载的比较变态,让人有点琢磨不透而已。对于箭头运算符,一般的左边为指针,不过这里 重载的话左边为一个类对象,注意看声明时在B中的,所以左边的那个就是B类型的变量,也就是b,而这个是单目运算符,于是重载之后返回的指针,照理说应该 是b-> –>a,因为左边做了操作之后返回的是指针,然后再访问a,不过应该是这里的编译器处理或者规定,所以就只剩下了一个->,从此处,我看出来 了重载的参数顺序,以及参数的个数。以前总是手动的加上个参数,或者少一个,现在将this算入其中,然后看是否为成员,就可以得到确定的数目,然后就可 以知道具体的调用顺序了。

分享到:
评论

相关推荐

    java基础知识点研究.txt

    1、赋值顺序不同 ++ i 是先加后赋值;i ++ 是先赋值后加;++i和i++都是分两步完成的。 因为++i 是后面一步才赋值的,所以它能够当作一个...形象的理解可以是i++先做别的事,再自己加1,++i先自己加1,再做别的事情。

    计算数_将简单的加减乘除括号公式解析成树的结构.zip

    使计算数_将简单的加减乘除括号公式解析成树的结构,用python解析简单的加减乘除括号表达式 成计算树,并以左值计算打印

    c-c++及数据结构基础视频教程完整版,最适合自学的c++基础

    第二部分C++基础目录 01_C++基础课程的安排和需要持之以恒的学习态度 02_简单的C++程序helloworld 03_用面向过程和面向对象...21_函数返回值是引用(当左值右值)_传智扫地僧源码及文档 01_上一次课程复习 02_指针的引用_

    解决三元运算符 报错“SyntaxError: can”t assign to conditional expression”

    expression是表达式,就是加减乘除等各种运算符号连接起来的式子(statement是语句,如if语句,while,复制语句等); 三目运算中表达式只能作为左值 修改后: a=1 b=1 x=1 if a==b else 0 print(x) [on true] if ...

    c++11智能指针解析——揭开底层面纱,完整理解智能指针.pdf

    c++11新加的unique_ptr, shared_ptr以及weak_ptr。 头⽂件:#include &lt;memory&gt; 1.auto_ptr auto_ptr是我第⼀个看的智能指针,也是标准库⾥的智能指针,有许多缺陷。 ⾸先看下结构: 从图中可以看书也是⼀个模板,使...

    指针数组和数组指针的区别.doc

    和代码中的char *test[]一样,同为字符指针,当你把参数传递过来的时候,事实上不是把数组内容传递过来,test的首地址传递了进来,由于array是指针,所以在内存中它在栈区,具有变量一样的性质,可以为左值,所以我们输出写成...

    传智播客扫地僧视频讲义源码

    17_信息系统框架集成第三方产品案例_加解密抽象类和加解密厂商类实现 18_信息系统框架集成第三方产品案例_集成测试加密厂商和socket厂商入围 19_信息系统框架集成第三方产品案例_集成框架变成类方式_传智扫地僧 20_...

    一维动态数组实现的矩阵类

    2、类型定死为double,原来作业是模板类,由于vc6对模版支持不好,另矩阵计算double类比较理想、整型几乎只能作加减 提供了多种初始化方式,int[]、float[]、double[]均可构造初始化,或则先构造出CVector再由C...

    自定义的矩阵类,内含源码与测试工程

    2、类型定死为double,原来作业是模板类,由于vc6对模版支持不好,另矩阵计算double类比较理想、整型几乎只能作加减 提供了多种初始化方式,int[]、float[]、double[]均可构造初始化,或则先构造出CVector再由C...

    C 语言编程常见问题解答.chm

    12 运算符的优先级总能保证是“自左至右”或“自右至左”的顺序吗? 1. 13 ++var和var++有什么区别? 1. 14 取模运算符(modulusoperator)“%”的作用是什么? 第2章 变量和数据存储 2. 1 变量存储在内存...

    C语言编程要点

    1.12 运算符的优先级总能保证是“自左至右”或“自右至左”的顺序吗? 17 1.13 ++var和var++有什么区别? 17 1.14 取模运算符(modulus operator)“%”的作用是什么? 17 第2章 变量和数据存储 18 2.1. 变量存储在...

    C语言FAQ 常见问题列表

    o 6.6 如果 NULL 定义成 #define NULL ((char *)0) 难道不就可以向函数传入不加转换的 NULL 了吗? o 6.7 如果 NULL 和 0 作为空指针常数是等价的, 那我到底该用哪一个呢? o 6.8 但是如果 NULL 的值改变了, 比如...

    你必须知道的495个C语言问题.pdf

    5.6 如果NULL定义成#define NULL((char *)0) ,不就可以向函数传入不加转换的NULL 了吗? 5.7 我的编译器提供的头文件中定义的NULL为0L。为什么? 5.8 NULL可以合法地用作函数指针吗? 5.9 如果NULL和0作为空指针...

    你必须知道的495个C语言问题

    5.6 如果NULL定义成#defineNULL((char*)0),不就可以向函数传入不加转换的NULL了吗? 5.7 我的编译器提供的头文件中定义的NULL为0L。为什么? 5.8 NULL可以合法地用作函数指针吗? 5.9 如果NULL和0作为空指针...

    《你必须知道的495个C语言问题》

    5.6 如果NULL定义成#define NULL((char *)0) ,不就可以向函数传入不加转换的NULL 了吗? 57 5.7 我的编译器提供的头文件中定义的NULL为0L。为什么? 57 5.8 NULL可以合法地用作函数指针吗? 57 5.9 如果NULL...

Global site tag (gtag.js) - Google Analytics