Skip to content

Latest commit

 

History

History
202 lines (164 loc) · 19.6 KB

10 Exceptions.md

File metadata and controls

202 lines (164 loc) · 19.6 KB

Chapter 10. Exceptions

Item 69: Use exceptions only for exceptional conditions

  • “Exceptions are, as their name implies, to be used only for exceptional conditions; they should never be used for ordinary control flow.”
    • “More generally, use standard, easily recognizable idioms in preference to overly clever techniques that purport to offer better performance.”
    • “Even if the performance advantage is real, it may not remain in the face of steadily improving platform implementations.”
    • “The subtle bugs and maintenance headaches that come from overly clever techniques, however, are sure to remain.”
  • “A well-designed API must not force its clients to use exceptions for ordinary control flow.”
    • “A class with a “state-dependent” method that can be invoked only under certain unpredictable conditions should generally have a separate “state-testing” method indicating whether it is appropriate to invoke the state-dependent method.”
      • “For example, the Iterator interface has the state-dependent method next and the corresponding state-testing method hasNext.”
    • “An alternative to providing a separate state-testing method is to have the state-dependent method return an empty optional (Item 55) or a distinguished value such as null if it cannot perform the desired computation.”
    • “If an object is to be accessed concurrently without external synchronization or is subject to externally induced state transitions, you must use an optional or distinguished return value, as the object’s state could change in the interval between the invocation of a state-testing method and its state-dependent method. Performance concerns may dictate that an optional or distinguished return value be used if a separate state-testing method would duplicate the work of the state-dependent method.”
    • “All other things being equal, a state-testing method is mildly preferable to a distinguished return value. It offers slightly better readability, and incorrect use may be easier to detect: if you forget to call a state-testing method, the state-dependent method will throw an exception, making the bug obvious; if you forget to check for a distinguished return value, the bug may be subtle. ”
  • “In summary, exceptions are designed for exceptional conditions. Don’t use them for ordinary control flow, and don’t write APIs that force others to do so.”

Item 70: Use checked exceptions for recoverable conditions and runtime exceptions for programming errors

  • “Java provides three kinds of throwables: checked exceptions, runtime exceptions, and errors.”
  • “The cardinal rule in deciding whether to use a checked or an unchecked exception is this: use checked exceptions for conditions from which the caller can reasonably be expected to recover.”
    • “By throwing a checked exception, you force the caller to handle the exception in a catch clause or to propagate it outward.”
    • “Each checked exception that a method is declared to throw is therefore a potent indication to the API user that the associated condition is a possible outcome of invoking the method.”
  • “There are two kinds of unchecked throwables: runtime exceptions and errors.”
    • “They are identical in their behavior: both are throwables that needn’t, and generally shouldn’t, be caught.”
    • “If a program does not catch such a throwable, it will cause the current thread to halt with an appropriate error message.”
  • “Use runtime exceptions to indicate programming errors.”
    • “The great majority of runtime exceptions indicate precondition violations. A precondition violation is simply a failure by the client of an API to adhere to the contract established by the API specification.”
  • “While the Java Language Specification does not require it, there is a strong convention that errors are reserved for use by the JVM to indicate resource deficiencies, invariant failures, or other conditions that make it impossible to continue execution. ”
    • “Therefore, all of the unchecked throwables you implement should subclass RuntimeException (directly or indirectly).”
    • “Not only shouldn’t you define Error subclasses, but with the exception of AssertionError, you shouldn’t throw them either.”
  • “API designers often forget that exceptions are full-fledged objects on which arbitrary methods can be defined. The primary use of such methods is to provide code that catches the exception with additional information concerning the condition that caused the exception to be thrown.”
    • “In the absence of such methods, programmers have been known to parse the string representation of an exception to ferret out additional information. ”

    • “This is extremely bad practice (Item 12).”

    • “Because checked exceptions generally indicate recoverable conditions, it’s especially important for them to provide methods that furnish information to help the caller recover from the exceptional condition.”

  • “To summarize, throw checked exceptions for recoverable conditions and unchecked exceptions for programming errors. When in doubt, throw unchecked exceptions. Don’t define any throwables that are neither checked exceptions nor runtime exceptions. Provide methods on your checked exceptions to aid in recovery.”

Item 71: Avoid unnecessary use of checked exceptions

  • “Many Java programmers dislike checked exceptions, but used properly, they can improve APIs and programs. Unlike return codes and unchecked exceptions, they force programmers to deal with problems, enhancing reliability. That said, overuse of checked exceptions in APIs can make them far less pleasant to use.”

  • “If a method throws a single checked exception, this exception is the sole reason the method must appear in a try block and can’t be used directly in streams.”

  • “The easiest way to eliminate a checked exception is to return an optional of the desired result type (Item 55).”

    • “Instead of throwing a checked exception, the method simply returns an empty optional.”
    • “The disadvantage of this technique is that the method can’t return any additional information detailing its inability to perform the desired computation.”
    • “Exceptions, by contrast, have descriptive types, and can export methods to provide additional information (Item 70).”
  • “You can also turn a checked exception into an unchecked exception by breaking the method that throws the exception into two methods, the first of which returns a boolean indicating whether the exception would be thrown.”

    • “This refactoring is not always appropriate, but where it is, it can make an API more pleasant to use.”
  • “In summary, when used sparingly, checked exceptions can increase the reliability of programs; when overused, they make APIs painful to use. If callers won’t be able to recover from failures, throw unchecked exceptions. If recovery may be possible and you want to force callers to handle exceptional conditions, first consider returning an optional. Only if this would provide insufficient information in the case of failure should you throw a checked exception.”

Item 72: Favor the use of standard exceptions

  • “Reusing standard exceptions has several benefits.”

    • “Chief among them is that it makes your API easier to learn and use because it matches the established conventions that programmers are already familiar with.”
    • “A close second is that programs using your API are easier to read because they aren’t cluttered with unfamiliar exceptions.”
    • “Last (and least), fewer exception classes means a smaller memory footprint and less time spent loading classes.”
  • “The most commonly reused exception type is IllegalArgumentException (Item 49). This is generally the exception to throw when the caller passes in an argument whose value is inappropriate.”

  • “Another commonly reused exception is IllegalStateException. This is generally the exception to throw if the invocation is illegal because of the state of the receiving object.”

  • “If a caller passes null in some parameter for which null values are prohibited, convention dictates that NullPointerException be thrown rather than IllegalArgumentException.”

  • “Similarly, if a caller passes an out-of-range value in a parameter representing an index into a sequence, IndexOutOfBoundsException should be thrown rather than IllegalArgumentException.”

  • “Another reusable exception is ConcurrentModificationException. It should be thrown if an object that was designed for use by a single thread (or with external synchronization) detects that it is being modified concurrently.”

  • “A last standard exception of note is UnsupportedOperationException. This is the exception to throw if an object does not support an attempted operation.”

  • “Do not reuse Exception, RuntimeException, Throwable, or Error directly.”

  • “If an exception fits your needs, go ahead and use it, but only if the conditions under which you would throw it are consistent with the exception’s documentation: reuse must be based on documented semantics, not just on name.”

  • “Also, feel free to subclass a standard exception if you want to add more detail (Item 75), but remember that exceptions are serializable (Chapter 12). That alone is reason not to write your own exception class without good reason.”

  • “Throw IllegalStateException if no argument values would have worked, otherwise throw IllegalArgumentException.”

Item 73: Throw exceptions appropriate to the abstraction

  • “It is disconcerting when a method throws an exception that has no apparent connection to the task that it performs.”
    • “This often happens when a method propagates an exception thrown by a lower-level abstraction.”
    • “Not only is it disconcerting, but it pollutes the API of the higher layer with implementation details.”
    • “If the implementation of the higher layer changes in a later release, the exceptions it throws will change too, potentially breaking existing client programs.”
  • “To avoid this problem, higher layers should catch lower-level exceptions and, in their place, throw exceptions that can be explained in terms of the higher-level abstraction.”
// Exception Translation
try {
    ... // Use lower-level abstraction to do our bidding
} catch (LowerLevelException e) {
    throw new HigherLevelException(...);
}
  • “A special form of exception translation called exception chaining is called for in cases where the lower-level exception might be helpful to someone debugging the problem that caused the higher-level exception. The lower-level exception (the cause) is passed to the higher-level exception, which provides an accessor method (Throwable’s getCause method) to retrieve the lower-level exception:”
// Exception Chaining
try {
    ... // Use lower-level abstraction to do our bidding
} catch (LowerLevelException cause) {
    throw new HigherLevelException(cause);
} 
  • “The higher-level exception’s constructor passes the cause to a chaining-aware superclass constructor, so it is ultimately passed to one of Throwable’s chaining-aware constructors, such as Throwable(Throwable):”
// Exception with chaining-aware constructor
class HigherLevelException extends Exception {
    HigherLevelException(Throwable cause) {
        super(cause);
    }
}
  • “While exception translation is superior to mindless propagation of exceptions from lower layers, it should not be overused.”
    • “Where possible, the best way to deal with exceptions from lower layers is to avoid them, by ensuring that lower-level methods succeed.”
  • “If it is impossible to prevent exceptions from lower layers, the next best thing is to have the higher layer silently work around these exceptions, insulating the caller of the higher-level method from lower-level problems.”
  • “In summary, if it isn’t feasible to prevent or to handle exceptions from lower layers, use exception translation, unless the lower-level method happens to guarantee that all of its exceptions are appropriate to the higher level. Chaining provides the best of both worlds: it allows you to throw an appropriate higher-level exception, while capturing the underlying cause for failure analysis (Item 75).”

Item 74: Document all exceptions thrown by each method

  • “A description of the exceptions thrown by a method is an important part of the documentation required to use the method properly.”
  • Always declare checked exceptions individually, and document precisely the conditions under which each one is thrown using the Javadoc @throws tag.”
    • “Don’t take the shortcut of declaring that a method throws some superclass of multiple exception classes that it can throw.”
  • “While the language does not require programmers to declare the unchecked exceptions that a method is capable of throwing, it is wise to document them as carefully as the checked exceptions.”
    • “A well-documented list of the unchecked exceptions that a method can throw effectively describes the preconditions for its successful execution.”
  • “It is particularly important that methods in interfaces document the unchecked exceptions they may throw. This documentation forms a part of the interface’s general contract and enables common behavior among multiple implementations of the interface.”
  • “Use the Javadoc @throws tag to document each exception that a method can throw, but do not use the throws keyword on unchecked exceptions.”
    • “The documentation generated by the Javadoc @throws tag without a corresponding throws clause in the method declaration provides a strong visual cue to the programmer that an exception is unchecked.”
  • “If an exception is thrown by many methods in a class for the same reason, you can document the exception in the class’s documentation comment rather than documenting it individually for each method.”
  • “In summary, document every exception that can be thrown by each method that you write. This is true for unchecked as well as checked exceptions, and for abstract as well as concrete methods. This documentation should take the form of @throws tags in doc comments. Declare each checked exception individually in a method’s throws clause, but do not declare unchecked exceptions. If you fail to document the exceptions that your methods can throw, it will be difficult or impossible for others to make effective use of your classes and interfaces.”

Item 75: Include failure-capture information in detail messages

  • “When a program fails due to an uncaught exception, the system automatically prints out the exception’s stack trace.”

    • “The stack trace contains the exception’s string representation, the result of invoking its toString method.”
    • “This typically consists of the exception’s class name followed by its detail message.”
    • “Frequently this is the only information that programmers or site reliability engineers will have when investigating a software failure. If the failure is not easily reproducible, it may be difficult or impossible to get any more information.”
  • “To capture a failure, the detail message of an exception should contain the values of all parameters and fields that contributed to the exception.”

    • “For example, the detail message of an IndexOutOfBoundsException should contain the lower bound, the upper bound, and the index value that failed to lie between the bounds.”
  • “One caveat concerns security-sensitive information.”

    • “Because stack traces may be seen by many people in the process of diagnosing and fixing software issues, do not include passwords, encryption keys, and the like in detail messages.”
  • “Lengthy prose descriptions of the failure are superfluous; the information can be gleaned by reading the documentation and source code.”

  • “The detail message of an exception should not be confused with a user-level error message, which must be intelligible to end users.”

/**
 * Constructs an IndexOutOfBoundsException.
 *
 * @param lowerBound the lowest legal index value
 * @param upperBound the highest legal index value plus one
 * @param index      the actual index value
 */
public IndexOutOfBoundsException(int lowerBound, int upperBound,
                                 int index) {
    // Generate a detail message that captures the failure
    super(String.format(
            "Lower bound: %d, Upper bound: %d, Index: %d",
            lowerBound, upperBound, index));

    // Save failure information for programmatic access
    this.lowerBound = lowerBound;
    this.upperBound = upperBound;
    this.index = index;
}
  • “It is more important to provide such accessor methods on checked exceptions than unchecked, because the failure-capture information could be useful in recovering from the failure.”

Item 76: Strive for failure atomicity

  • “After an object throws an exception, it is generally desirable that the object still be in a well-defined, usable state, even if the failure occurred in the midst of performing an operation. This is especially true for checked exceptions, from which the caller is expected to recover.”
  • “Generally speaking, a failed method invocation should leave the object in the state that it was in prior to the invocation.”
    • “A method with this property is said to be failure-atomic.”
  • “For methods that operate on mutable objects, the most common way to achieve failure atomicity is to check parameters for validity before performing the operation (Item 49). This causes most exceptions to get thrown before object modification commences.”
  • “A closely related approach to achieving failure atomicity is to order the computation so that any part that may fail takes place before any part that modifies the object.”
  • “A third approach to achieving failure atomicity is to perform the operation on a temporary copy of the object and to replace the contents of the object with the temporary copy once the operation is complete.”
  • “A last and far less common approach to achieving failure atomicity is to write recovery code that intercepts a failure that occurs in the midst of an operation, and causes the object to roll back its state to the point before the operation began.”
  • “In summary, as a rule, any generated exception that is part of a method’s specification should leave the object in the same state it was in prior to the method invocation. Where this rule is violated, the API documentation should clearly indicate what state the object will be left in. Unfortunately, plenty of existing API documentation fails to live up to this ideal.”

Item 77: Don’t ignore exceptions

  • “It is easy to ignore exceptions by surrounding a method invocation with a try statement whose catch block is empty:”
// Empty catch block ignores exception - Highly suspect!
try {
    ...
} catch (SomeException e) {
}
  • An empty catch block defeats the purpose of exceptions, which is to force you to handle exceptional conditions.”
  • “If you choose to ignore an exception, the catch block should contain a comment explaining why it is appropriate to do so, and the variable should be named ignored:”
Future<Integer> f = exec.submit(planarMap::chromaticNumber);
int numColors = 4; // Default; guaranteed sufficient for any map
try {
    numColors = f.get(1L, TimeUnit.SECONDS);
} catch (TimeoutException | ExecutionException ignored) {
    // Use default: minimal coloring is desirable, not required
}