第39课 逗号操作符的分析

Posted -glb

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了第39课 逗号操作符的分析相关的知识,希望对你有一定的参考价值。

逗号操作符
逗号操作符(,)可以构成逗号表达式
-逗号表达式用于将多个子表达式连接为一个表达式
-逗号表达式的值为最后一个子表达式的值
-逗号表达式中的前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.重载后无法严格从左向有计算表达式

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

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

以上是关于第39课 逗号操作符的分析的主要内容,如果未能解决你的问题,请参考以下文章

第18课 三目运算符和逗号表达式

Oracle开发者中级第4课(分析函数)实验

第17课 ++和--操作符分析

第17课 ++和--操作符分析

第22课 条件编译使用分析

python 第22课 列表 内存分析 列表常用操作