End points and reachability
Every statement has an end point. In intuitive terms, the end point of a statement is the location that immediately follows the statement. The execution rules for composite statements (statements that contain embedded statements) specify the action that is taken when control reaches the end point of an embedded statement. For example, when control reaches the end point of a statement in a block, control is transferred to the next statement in the block. If a statement can possibly be reached by execution, the statement is said to be reachable. Conversely, if there is no possibility that a statement will be executed, the statement is said to be unreachable. In the example void F() { the second invocation of Console.WriteLine is unreachable because there is no possibility that the statement will be executed. A warning is reported if the compiler determines that a statement is unreachable. It is specifically not an error for a statement to be unreachable. To determine whether a particular statement or end point is reachable, the compiler performs flow analysis according to the reachability rules defined for each statement. The flow analysis takes into account the values of constant expressions (§7.19) that control the behavior of statements, but the possible values of non-constant expressions are not considered. In other words, for purposes of control flow analysis, a non-constant expression of a given type is considered to have any possible value of that type. In the example void F() { the boolean expression of the if statement is a constant expression because both operands of the == operator are constants. As the constant expression is evaluated at compile-time, producing the value false, the Console.WriteLine invocation is considered unreachable. However, if i is changed to be a local variable void F() { the Console.WriteLine invocation is considered reachable, even though, in reality, it will never be executed. The block of a function member is always considered reachable. By successively evaluating the reachability rules of each statement in a block, the reachability of any given statement can be determined. In the example void F(int x) { the reachability of the second Console.WriteLine is determined as follows: · The first Console.WriteLine expression statement is reachable because the block of the F method is reachable. · The end point of the first Console.WriteLine expression statement is reachable because that statement is reachable. · The if statement is reachable because the end point of the first Console.WriteLine expression statement is reachable. · The second Console.WriteLine expression statement is reachable because the boolean expression of the if statement does not have the constant value false. There are two situations in which it is a compile-time error for the end point of a statement to be reachable: · Because the switch statement does not permit a switch section to “fall through” to the next switch section, it is a compile-time error for the end point of the statement list of a switch section to be reachable. If this error occurs, it is typically an indication that a break statement is missing. · It is a compile-time error for the end point of the block of a function member that computes a value to be reachable. If this error occurs, it typically is an indication that a return statement is missing. Blocks A block permits multiple statements to be written in contexts where a single statement is allowed. block: A block consists of an optional statement-list (§8.2.1), enclosed in braces. If the statement list is omitted, the block is said to be empty. A block may contain declaration statements (§8.5). The scope of a local variable or constant declared in a block is the block. Within a block, the meaning of a name used in an expression context must always be the same (§7.6.2.1). A block is executed as follows: · If the block is empty, control is transferred to the end point of the block. · If the block is not empty, control is transferred to the statement list. When and if control reaches the end point of the statement list, control is transferred to the end point of the block. The statement list of a block is reachable if the block itself is reachable. The end point of a block is reachable if the block is empty or if the end point of the statement list is reachable. A block that contains one or more yield statements (§8.14) is called an iterator block. Iterator blocks are used to implement function members as iterators (§10.14). Some additional restrictions apply to iterator blocks: · It is a compile-time error for a return statement to appear in an iterator block (but yield return statements are permitted). · It is a compile-time error for an iterator block to contain an unsafe context (§18.1). An iterator block always defines a safe context, even when its declaration is nested in an unsafe context.
|