VLA和操作数大小的副作用
我知道sizeof
从不评估其操作数,除非在所述操作数是VLA的特定情况下。 或者,我以为我知道。
void g(int n) { printf("g(%d)\n", n); } int main(void) { int i = 12; char arr[i]; // VLA (void)sizeof *(g(1), &arr); // Prints "g(1)" (void)sizeof (g(2), arr); // Prints nothing return 0; }
到底是怎么回事?
以防万一,这是在Coliru上用GCC 5.1编译的。
在发布之前我似乎应该三思而后行,因为在我这样做之后它就让我感到震惊。
我对sizeof
如何与VLA交互的理解实际上是正确的,如下面的引用所证实的那样(感谢@this!):
6.5.3.4
sizeof
和_Alignof
运算符
如果操作数的类型是可变长度数组类型,则计算操作数; 否则,不评估操作数,结果是整数常量
这不是造成这种令人惊讶(对我而言)行为的原因。
(void)sizeof (g(2), arr);
在(g(2), arr)
子表达式中,逗号运算符触发arr
的数组到指针衰减。 因此, sizeof
的操作数不再是VLA,而是普通的char*
,它回退到不评估其操作数。
显然,这种行为在C ++中已被改变,其中逗号运算符将不再衰减数组。