zoukankan      html  css  js  c++  java
  • 第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.重载后无法严格从左向有计算表达式

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

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

  • 相关阅读:
    字符串函数---atof()函数具体解释及实现(完整版)
    curl的简单使用
    [7] 算法之路
    springMVC3.0(文件上传,@RequestMapping加參数,@SessionAttributes,@ModelAttribute,转发,重定向,数值获取,传參,ajax,拦截器)
    hdu 1754 I Hate It 线段树 点改动
    经典的7种排序算法 原理C++实现
    自己定义View实现水平滚动控件
    centos编译ffmpeg x264
    工作脚本处理文本
    A*寻路算法
  • 原文地址:https://www.cnblogs.com/-glb/p/11939078.html
Copyright © 2011-2022 走看看