IT博客汇
  • 首页
  • 精华
  • 技术
  • 设计
  • 资讯
  • 扯淡
  • 权利声明
  • 登录 注册

    C陷阱与缺陷之求值顺序

    TacuLee发表于 2016-01-14 05:32:46
    love 0

    求值顺序与运算符的优先级问题完全是另一码事。运算符优先级是关于诸如表达式

    a+b*c

    应该被解释成

    a+(b*c)

    而不是

    (a+c)*c

    的这样一类规则。求值顺序是另一类规则,可以保证像下面的语句

    if(count !=0&&sum/count<smallaverage)
           printf(“average < %gn”, smallaverage);

    即使当变量count为0时,也不会产生一个“用0作除数”的错误。

    C语言中的某些运算符是以一种已知、规定的顺序来对其操作数进行求值,而另外一些则不是这样。例如,考虑下面的表达式:

           a<b&&c<d

    C语言的定义中说明a<b应当首先被求值。如果a确实小于b,此时必须进一步对c<d求值,以确定整个表达式的值。但是,如果a大于或等b,则无需对c<d求值,表达式肯定为假。

    另外,要对a<b求值,编译器可能先对a求值,也可能先对b求值,在某些机器上甚至有可能对它们并行求值。

    C语言中只有四个运算符(&&、||、?;和,)存在规定的求值顺序。运算符&&和运算符||首先对左侧操作数求值,只有需要时才对右侧操作数求值。运算符?;有三个操作数:在a?b:c中,操作数a首先被求值,根据a的值再求操作数b或c的值。而逗号运算符,首先对左侧操作数求值,然后该值被“丢弃”,再对右侧操作数求值。

    注:分隔函数参数的逗号并非逗号运算符。例如,x和y在函数f(x,y)中的求值顺序是未定义的,而在函数g((x,y))中却是确定的先x后y的顺序。在后一个例子中,函数g只有一个参数。这个参数的值是这样求得的先对x求值,然后x的值被“丢弃”,接着求y的值。

    C语言中其他所有运算符对其操作数求值的顺序是未定义的。特别地,赋值运算符并不保证任何求值顺序。

    运算符&&和运算符||对于保证检查操作正确的顺序执行至关重要。例如,在语句

    if(y!=&&x/y?tolerance)
           complain();

    中,就必须保证仅当y非0时才对x/y求值。

    下面这种从数组x中复制前n个元素到数组y中的做法是不正确的,因为它对求值顺序作了太多的假设。

    i=0;
    while(i<n)
           y[i]=x[i++];

    问题出在呢?上面的代码假设y[i]的地址将在i的自增操作执行之前被求值,这一点并没任何保证!在C语言的某些实现上,有可能i自增之前被求值;而在另外一些实现上,有可能与此相反。同样道理,下面这种版本的写法,与前类似,也不正确:

    i=0;
    while(i<n)
           y[i++]=x[i];

    另一方面,下面这种写法却能正确工作。

    i=0;
    while(i<n){
            y[i]=x[i];
            i++;
    }

    当然这种写法也可以简写为:

    for(ji=0;i<n;i++)
           y[i]=x[i];

    未经允许不得转载:TacuLee » C陷阱与缺陷之求值顺序



沪ICP备19023445号-2号
友情链接