This page covers the following topics:
In Java SE 7 and later, a single catch
block can handle more than one type of exception. This feature can reduce code duplication and lessen the temptation to catch an overly broad exception.
Consider the following example, which contains duplicate code in each of the catch
blocks:
catch (IOException ex) { logger.log(ex); throw ex; catch (SQLException ex) { logger.log(ex); throw ex; }
In releases prior to Java SE 7, it is difficult to create a common method to eliminate the duplicated code because the variable ex
has different types.
The following example, which is valid in Java SE 7 and later, eliminates the duplicated code:
catch (IOException|SQLException ex) { logger.log(ex); throw ex; }
The catch
clause specifies the types of exceptions that the block can handle, and each exception type is separated with a vertical bar (|
).
Note: If a catch
block handles more than one exception type, then the catch
parameter is implicitly final
. In this example, the catch
parameter ex
is final
and therefore you cannot assign any values to it within the catch
block.
Bytecode generated by compiling a catch
block that handles multiple exception types will be smaller (and thus superior) than compiling many catch
blocks that handle only one exception type each. A catch
block that handles multiple exception types creates no duplication in the bytecode generated by the compiler; the bytecode has no replication of exception handlers.
The Java SE 7 compiler performs more precise analysis of rethrown exceptions than earlier releases of Java SE. This enables you to specify more specific exception types in the throws
clause of a method declaration.
Consider the following example:
static class FirstException extends Exception { } static class SecondException extends Exception { } public void rethrowException(String exceptionName) throws Exception { try { if (exceptionName.equals("First")) { throw new FirstException(); } else { throw new SecondException(); } } catch (Exception e) { throw e; } }
This examples's try
block could throw either FirstException
or SecondException
. Suppose you want to specify these exception types in the throws
clause of the rethrowException
method declaration. In releases prior to Java SE 7, you cannot do so. Because the exception parameter of the catch
clause, e
, is type Exception
, and the catch block rethrows the exception parameter e
, you can only specify the exception type Exception
in the throws
clause of the rethrowException
method declaration.
However, in Java SE 7, you can specify the exception types FirstException
and SecondException
in the throws
clause in the rethrowException
method declaration. The Java SE 7 compiler can determine that the exception thrown by the statement throw e
must have come from the try
block, and the only exceptions thrown by the try
block can be FirstException
and SecondException
. Even though the exception parameter of the catch
clause, e
, is type Exception
, the compiler can determine that it is an instance of either FirstException
or SecondException
:
public void rethrowException(String exceptionName) throws FirstException, SecondException { try { // ... } catch (Exception e) { throw e; } }
This analysis is disabled if the catch
parameter is assigned to another value in the catch
block. However, if the catch parameter is assigned to another value, you must specify the exception type Exception
in the throws
clause of the method declaration.
In detail, in Java SE 7 and later, when you declare one or more exception types in a catch
clause, and rethrow the exception handled by this catch
block, the compiler verifies that the type of the rethrown exception meets the following conditions:
try
block is able to throw it.catch
blocks that can handle it.catch
clause's exception parameters.The Java SE 7 compiler allows you to specify the exception types FirstException
and SecondException
in the throws
clause in the rethrowException
method declaration because you can rethrow an exception that is a supertype of any of the types declared in the throws
.
In releases prior to Java SE 7, you cannot throw an exception that is a supertype of one of the catch
clause's exception parameters. A compiler from a release prior to Java SE 7 generates the error, "unreported exception Exception
; must be caught or declared to be thrown" at the statement throw e
. The compiler checks if the type of the exception thrown is assignable to any of the types declared in the throws
clause of the rethrowException
method declaration. However, the type of the catch parameter e
is Exception
, which is a supertype, not a subtype, of FirstException
andSecondException
.