what's wrong with declaring a variable inside if's condition?

淺唱寂寞╮ 提交于 2019-12-04 22:49:52

You can declare a variable in the if statement in C++ but it is restricted to be used with direct initialization and it needs to convert to a Boolean value:

if (int i = f()) { ... }

C++ doesn't have anything which could be described as "declaration expression", i.e. [sub-] expressions declaring a variable.

Actually, I just looked up the clause in the standard and both forms of initialization are supported according to 6.4 [stmt.select] paragraph 1:

...
condition:
   expression
   attribute-specifier-seqopt decl-specifier-seq declarator = initializer-clause
   attribute-specifier-seqopt decl-specifier-seq declarator braced-init-list
...

That is, it is also be possible to write:

if (int i{f()}) { ... }

Obviously, this only works in C++2011 because C++2003 doesn't have brace-initialization.

There's a problem with scope.

Consider the following code:

if ((int a = foo1()) || (int b = foo2()))
{
    bar(b);
}

Is b declared inside the block? What if foo1() returns true?

You can declare a variable in an if statement (or in for or while), but only in the outer parenthesis block and it needs to be able to be converted to bool.

Your guess is basically right, it's not allowed because

(int i = 42;)

is not a valid declaration with initialization.

You need one additional line,

Mymap::iterator it;
if ( (it = m.find(name)) != m.end())
    return it->second;

but then it's better to write

Mymap::iterator it = m.find(name);
if ( it != m.end() ) 
    return it->second;

You can put the return line after the if, if you really want this line back, at least for me this doesn't harm readability, but others might see that different.

If you really, really want to declare an iterator and use the it as bool in an if condition, you could do

if ( struct { int it; operator bool() { return it != m.end; } } s = { m.find(name) } )
    return s.it->second;

but I would consider this harmful ;-)

It's true that you can't write

if ( (int i=f()) == 0)

but you can perfectly write

if ( int i=f())

So you can use the && operator to perform both operations in one statement like

if ( int i=1 && (i=f()) == 0)

i should be initialized with any value other than 0, and it should be the first condition if your compiler applies left-to-right evaluation.

But unfortunately, that's not applicable in case of iterators as your second example asks.

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!