Perhaps I am getting rusty (have been writing in Python recently).
Why does this not compile?
if ( (int i=f()) == 0)
without the ()
around the int i=f()
I get another, much more reasonable error of i
is not being boolean. But that's why I wanted the parentheses in the first place!
My guess would be that using the parentheses makes it into an expression, and that declaration statements are not allowed in an expression. Is it so? And if yes, is it one of the C++'s syntax quirks?
BTW, I was actually trying to do this:
if ( (Mymap::iterator it = m.find(name)) != m.end())
return it->second;
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.