第39课 逗号操作符的分析

痴心易碎 提交于 2019-11-28 15:11:22

逗号操作符
逗号操作符(,)可以构成逗号表达式
-逗号表达式用于将多个子表达式连接为一个表达式
-逗号表达式的值为最后一个子表达式的值
-逗号表达式中的前N-1个子表达式可以没有返回值
-逗号表达式按照从左向右的顺序计算每个子表达式的值
exp1, exp2, exp3, ... , expN

逗号表达式的实例分析 

#include <iostream>
#include <string>

using namespace std;

void func(int i)
{
    cout << "func() : i = " << i << endl;
}

int main()
{   
    int a[3][3] = {
        (0, 1, 2),
        (3, 4, 5),
        (6, 7, 8)
    };
    
    int i = 0;
    int j = 0;
    
    while( i < 5 )    
        func(i),
    
    i++;
        
    for(i=0; i<3; i++)
    {
        for(j=0; j<3; j++)
        {
            cout << a[i][j] << endl;
        }
    }
    
    (i, j) = 6;
    
    cout << "i = " << i << endl;
    cout << "j = " << j << endl;

    return 0;
}

 

 

 程序的运行结果,会让你觉得有点诧异。

 while( i < 5 )    
        func(i),
    
    i++;如果你不仔细看,还以为是死循环呢,即while( i < 5)  func(i);注意哦,原程序这个地方不是分号,而是逗号。如果是分号,就是死循环。原程序就相当于这样:while( i < 5 ){  func(i);  i++;}问题2:打印结果为什么是2 5 8 ,然后全是0.为什么会这样,难道不应该0 1 2 3 4 5 6 7 8
int a[3][3] = {
        (0, 1, 2),
        (3, 4, 5),
        (6, 7, 8)
    };
(0, 1, 2)这个地方是个括号,因此它就是一个逗号表达式,逗号表达式的值为最后一个表达式的值,即为2
(3, 4, 5)它为5
(6, 7, 8)它的值为8.下面的这种写法是赋值:
int a[3][3] = {
        {0, 1, 2},
        {3, 4, 5},
        {6, 7, 8}
    };这样写,就是打印出 0 1 2 3 4 5 6 7 8

重载逗号操作符

在C++中重载逗号操作符是合法的
使用全局函数对逗号操作符进行重载
重载函数的参数必须有一个是类类型
重载函数的返回值类型必须是引用

Class& operator , (const Class& a, const Class& b)
{
  return const_cast<Class&>(b); //返回的是第二个参数,使用了一个强制类型转换const_cast,将b的const属性去掉。因为我们返回的是Class的引用,而不是const引用。

}

#include <iostream>
#include <string>

using namespace std;

class Test
{
private:
    int mValue;
public:
    Test(int i)
    {
        mValue = i;
    }
    int value()
    {
        return mValue;
    }

};
Test& operator , (const Test& a, const Test& b)
{
    return const_cast<Test&>(b);
}
Test func(Test& i)
{
    cout << "func() : i = " << i.value() << endl;

    return i;
}

int main()
{
    Test t1(0);
    Test t2(1);

    //Test tt = (t1,t2); //等价于Test tt = t2
    Test tt = (func(t1), func(t2));  //等价于Test tt = func(t2)  //Test tt = operator,(func(t1),func(t2));实际上就是通过这种函数调用的形式实现。
    cout << tt.value() << endl;
    return 0;
}

程序运行结果:

 

 感觉程序运行不太对劲,这里的逗号表达式并没有从左到右逐个的调用,而是从右往左执行的。

虽然最后的结果是我们期望的,但是中间过程出错了。说明逗号表达式被重载之后,会出现问题,即中间过程发生了变化,这是不允许的,因为违背了逗号表达式的原生语义。

问题的本质分析
1.C++通过函数调用扩展操作符的功能
2.进入函数体前必须完成所有参数的计算
3.函数参数的计算次序是不定的 (无法从左向右计算了)
4.重载后无法严格从左向有计算表达式

工程中千万不要重载逗号表达式

小结:
逗号表达式从左向右顺序计算每个子表达式的值
逗号表达式的值为最后一个子表达式的值
操作符重载无法完全实现逗号操作符的原生意义
工程开发中不要重载逗号操作符

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!