The try statement provides a mechanism for catching exceptions that occur during execution of a block.
Furthermore, the try statement provides the ability to specify a block of code that is always executed when control leaves the try statement. try-statement : tryblockcatch-clausestryblockfinally-clausetryblockcatch-clausesfinally-clausecatch-clauses : specific-catch-clausesgeneral-catch-clausespecific-catch-clausesgeneral-catch-clausespecific-catch-clauses : specific-catch-clausespecific-catch-clausesspecific-catch-clausespecific-catch-clause : catch(class-typeidentifier)blockgeneral-catch-clause : catchblockfinally-clause : finallyblock
There are three possible forms of try statements: A try block followed by one or more catch blocks. A try block followed by a finally block. A try block followed by one or more catch blocks followed by a finally block.
When a catch clause specifies a class-type, the type must be System.Exception or a type that derives from System.Exception.
When a catch clause specifies both a class-type and an identifier, an exception variable of the given name and type is declared. The exception variable corresponds to a local variable with a scope that extends over the catch block. During execution of the catch block, the exception variable represents the exception currently being handled. For purposes of definite assignment checking, the exception variable is considered definitely assigned in its entire scope.
Unless a catch clause includes an exception variable name, it is impossible to access the exception object in the catch block.
A catch clause that specifies neither an exception type nor an exception variable name is called a general catch clause. A try statement can only have one general catch clause, and if one is present it must be the last catch clause.
[Note: Some environments, especially those supporting multiple languages, may support exceptions that are not representable as an object derived from System.Exception, although such an exception could never be generated by C# code. In such an environment, a general catch clause might be used to catch such an exception. Thus, a general catch clause is semantically different from one that specifies the type System.Exception, in that the former may also catch exceptions from other languages. end note]
In order to locate a handler for an exception, catch clauses are examined in lexical order. A compile-time error occurs if a catch clause specifies a type that is the same as, or is derived from, a type that was specified in an earlier catch clause for the same try. [Note: Without this restriction, it would be possible to write unreachable catch clauses. end note]
Within a catch block, a throw statement (15.9.5) with no expression can be used to re-throw the exception that was caught by the catch block. Assignments to an exception variable do not alter the exception that is re-thrown.
[Example: In the example the method F catches an exception, writes some diagnostic information to the console, alters the exception variable, and re-throws the exception. The exception that is re-thrown is the original exception, so the output produced is:
If the first catch block had thrown e instead of rethrowing the current exception, the output produced would be as follows: end example]
It is a compile-time error for a break, continue, or goto statement to transfer control out of a finally block. When a break, continue, or goto statement occurs in a finally block, the target of the statement must be within the same finally block, or otherwise a compile-time error occurs.
It is a compile-time error for a return statement to occur in a finally block.
A try statement is executed as follows: Control is transferred to the try block. When and if control reaches the end point of the try block: If the try statement has a finally block, the finally block is executed. Control is transferred to the end point of the try statement.
If an exception is propagated to the try statement during execution of the try block: The catch clauses, if any, are examined in order of appearance to locate a suitable handler for the exception. The first catch clause that specifies the exception type or a base type of the exception type is considered a match. A general catch clause is considered a match for any exception type. If a matching catch clause is located: If the matching catch clause declares an exception variable, the exception object is assigned to the exception variable. Control is transferred to the matching catch block. When and if control reaches the end point of the catch block: If the try statement has a finally block, the finally block is executed. Control is transferred to the end point of the try statement.
If an exception is propagated to the try statement during execution of the catch block: If the try statement has a finally block, the finally block is executed. The exception is propagated to the next enclosing try statement.
If the try statement has no catch clauses or if no catch clause matches the exception: If the try statement has a finally block, the finally block is executed. The exception is propagated to the next enclosing try statement.
The statements of a finally block are always executed when control leaves a try statement. This is true whether the control transfer occurs as a result of normal execution, as a result of executing a break, continue, goto, or return statement, or as a result of propagating an exception out of the try statement.
If an exception is thrown during execution of a finally block, the exception is propagated to the next enclosing try statement. If another exception was in the process of being propagated, that exception is lost. The process of propagating an exception is discussed further in the description of the throw statement (15.9.5).
The try block of a try statement is reachable if the try statement is reachable.
A catch block of a try statement is reachable if the try statement is reachable.
The finally block of a try statement is reachable if the try statement is reachable.
The end point of a try statement is reachable if both of the following are true: The end point of the try block is reachable or the end point of at least one catch block is reachable. If a finally block is present, the end point of the finally block is reachable.