编译器如何评估 C 中的条件

How does the compiler evaluate a condition in C

我有一个关于编译器如何计算 c 中的 'AND' 条件的问题。

说,我写这样的声明

if( (today is Thursday) && (Month is July) )
{
 //do something
}

假设今天不是星期四,但月份确实是七月。

编译器是否检查这两个条件,并执行 1&0 == 0? 或者一旦它看到今天不是星期四,它就会跳过并且甚至懒得检查月份条件,因为它无关紧要。

我正在使用以下 gcc

Using built-in specs.
COLLECT_GCC=gcc
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/4.6/lto-wrapper
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.6.3-1ubuntu5' --with-bugurl=file:///usr/share/doc/gcc-4.6/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.6 --enable-shared --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.6 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-gnu-unique-object --enable-plugin --enable-objc-gc --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.6.3 (Ubuntu/Linaro 4.6.3-1ubuntu5) 

&& 运算符在 C 下短路。这意味着如果评估的第一个条件失败并且足以决定表达式的结果,它将在那里终止。

根据 C 标准(6.5.13 逻辑与运算符)

4 Unlike the bitwise binary & operator, the && operator guarantees left-to-right evaluation; if the second operand is evaluated, there is a sequence point between the evaluations of the first and second operands. If the first operand compares equal to 0, the second operand is not evaluated.

至于逻辑或运算符(6.5.14逻辑或运算符)则

4 Unlike the bitwise | operator, the || operator guarantees left-to-right evaluation; if the second operand is evaluated, there is a sequence point between the evaluations of the first and second operands. If the first operand compares unequal to 0, the second operand is not evaluated.

使用此图表:

http://www.swansontec.com/sopc.html

您会看到 && 运算符是从左到右求值的。

一旦在 && 中遇到 'false' 表达式,则执行 'shortcuts' 余下的计算。

所以如果第一个表达式为假,则不计算第二个表达式

正如其他人已经说过的那样 && 是一个短路运算符,在 && 的情况下,一旦结果已知就会终止求值(这发生在 L-> 任何操作数求值为 zero/false 这是停止进一步评估的充分条件,因为 0 && a = 0)

很快你的伪代码:

if( (today is Thursday) && (Month is July) )
{
 //do something
}

相当于:

if (today is Thursday)
{
    if (Month is July)
    {
        //do something
    }
}