简单描述问题,请看下面的代码:
int main()
{
int a=123;
({if (a) a=0;});
return 0;
}
我从[-Wsequence-point]
收到了这个警告Line 4: warning: operation on 'a' may be undefined
我的g ++版本是4.4.5
我会很感激能解释这个简单问题的人。
顺便说一下,您可以在this中文网站的#7中找到原始程序和原始问题(不是必需的)
UPD1:
虽然将代码更改为({if(a) a=0; a;})
可以避免警告,但我发现问题的真正原因可能不是The last thing in the compound statement should be an expression followed by a semicolon
。
因为该纪录片还说If you use some other kind of statement last within the braces, the construct has type void, and thus effectively no value
。
一个例子可以显示出来:
int main()
{
int a=123, b;
({;});
({if (a) b=0;});
return 0;
}
此代码无警告! 所以我认为真正的原因是序列点。
请帮忙!
UPD2:
很抱歉@AndyProwl没有接受他在UPD1之前接受的答案。按照他的建议,我可能会问一个新问题(UPD1是一个与原始问题不同的新问题)。我会再次接受他的回答,因为无论如何它肯定会避免警告。:)
如果我决定提出新问题,我会更新此问题以添加链接。
答案 0 :(得分:5)
根据C ++语法,表达式(除了lambda表达式,但这是一个不同的故事)不能包含语句 - 包括块语句。因此,我会说你的代码格式不正确,如果GCC编译它,这意味着这是一个(奇怪的)编译器扩展。
你应该参考编译器的参考,找出它给出的语义(或者没有给出,如错误消息似乎暗示的那样)。
修改强>
As pointed out by Shafik Yaghmour in the comments,这似乎是一个GNU扩展。根据{{3}},这个“语句表达式”的值应该是块中最后一个语句的值,它应该是一个表达式语句:
复合语句中的最后一件事应该是一个后跟分号的表达式;此子表达式的值用作整个构造的值。 (如果你在大括号中最后使用了一些其他类型的语句,那么构造的类型为void,因此实际上没有值。)
由于示例中的块不包含表达式语句作为最后一个语句,因此GCC不知道如何评估该“语句表达式”(不要与“表达式语句”混淆 - 这应该是最后出现在声明表达)。
因此,为防止GCC抱怨,您应该执行以下操作:
({if (a) a=0; a;});
// ^^
但老实说,我不明白为什么人们会在C ++中需要这个东西。