4.2 DeclarationsA declaration can appear anywhere a statement appears, and certain statements permit additional declarations within those statements. Declarations made in a substatement (of a selection or loop statement) are limited in scope to the substatement, even if the substatement is not a compound statement. For example, the following statement: while ( test( ) ) int x = init( ); is equivalent to: while ( test( ) ) { int x = init( ); } The first example uses a declaration as the entire loop body, and the second uses a compound statement (enclosing the loop body in curly braces). In both cases, though, the scope of x is limited to the body of the while loop. 4.2.1 Declaration StatementsA simple declaration can appear wherever a statement can be used. You can declare an object, a type, or a namespace alias. You can also write a using declaration or using directive. You can declare a function, but not define a function, although there is rarely any reason to declare a function locally. You cannot define a namespace or declare a template. In traditional C programming, declarations appear at the start of each block or compound statement. In C++ (and in the C99 standard), declarations can appear anywhere a statement can, which means you can declare variables close to where they are used. Example 4-1 shows examples of how declarations can be mixed with statements. Example 4-1. Mixing declarations and statements#include <cctype> #include <cstddef> #include <iomanip> #include <iostream> #include <ostream> #include <string> // Count lines, words, and characters in the standard input. int main( ) { unsigned long num_lines, num_words, num_chars; num_lines = num_words = num_chars = 0; using namespace std; string line; while (getline(cin, line)) { ++num_lines; num_chars += line.length( ) + 1; bool in_word = false; for (size_t i = 0; char c = line[i]; ++i) if (isspace(static_cast<unsigned char>(c))) { if (in_word) ++num_words; in_word = false; } else if (! in_word) in_word = true; if (in_word) ++num_words; } cout << right << setw(10) << num_lines << setw(10) << num_words << setw(10) << num_chars << '\n'; } Sometimes a construct can look like an expression statement or a declaration. These ambiguities are resolved in favor of declarations. Example 4-2 shows some declarations that look like they might be expressions. Example 4-2. Declarations that seem like expressionsclass cls { public: cls( ); cls(int x); }; int x; int* y = &x; int main( ) { // The following are unambiguously expressions, constructing instances of cls. cls(int(x)); cls(*static_cast<int*>(y)); // Without the redundant casts, though, they would look like declarations, not // expressions. cls(x); // Declares a variable x cls(*y); // Declares a pointer y } 4.2.2 Condition DeclarationsThe for, if, switch, and while statements permit a declaration within each statement's condition: if (int x = test_this(a, b)) { cout << x; } If the condition contains a declaration, the scope of the declared name extends to the end of the entire statement. In particular, a name declared in the condition of an if statement is visible in the else part of the statement. In a loop, the condition object is created and destroyed for each iteration of the loop. The name cannot be redeclared in the immediate substatement (but can be redeclared in a nested statement). The name is not visible in subsequent statements. For example: if (derived* d = dynamic_cast<derived*>(b)) { d->derived_only_func( ); } else { assert(d == NULL); // Same d as above double d; // Error: can't redeclare d if (d == 0) int d; // Valid: inner block } cout << d; // Invalid: d no longer in scope Like the if, switch, and while statements, a for loop permits a declaration in its condition. Unlike those other statements, it also allows a declaration to initialize the loop. Both declarations are in the same scope. See Section 4.5.2 for details. The syntax for a condition declaration is: type-specifiers declarator = expression This syntax is similar to the syntax of a simple object declaration. In this case, the initializer is required (without it, the condition would not have a value), and only one declarator is permitted. See Chapter 2 for more information about type specifiers and declarators. |