• 第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.重载后无法严格从左向有计算表达式

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

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

  • 相关阅读:
    鲁迅说过搜索引擎
    下载github上文件与release的安装包-解决s3.amazonaws.com问题
    作业九----DFA最小化
    作业八----非确定的自动机NFA确定化为DFA
    作业七----正规式到正规文法与自动机
    作业六----正规文法与正规式
    第五次作业----词法分析程序的设计与实现
    第四次作业
    作业三
    2.文法和语言
  • 原文地址:https://www.cnblogs.com/-glb/p/11939078.html
Copyright © 2020-2023  润新知