Flashcards for topic Exceptions
What's wrong with the following loop pattern and why should it be avoided?
try { int i = 0; while(true) range[i++].climb(); } catch (ArrayIndexOutOfBoundsException e) { }
This code abuses exceptions for ordinary control flow - a serious anti-pattern because:
The proper alternative is a standard loop:
for (Mountain m : range) m.climb();
Exceptions should only be used for exceptional conditions, not regular control flow.
What is the "cardinal rule" for deciding between checked and unchecked exceptions, and what factors should influence this decision?
Cardinal rule: Use checked exceptions for conditions from which the caller can reasonably be expected to recover.
Decision factors:
Use checked exceptions when:
Use runtime exceptions when:
If it's unclear whether recovery is possible, favor unchecked exceptions.
Remember: Checked exceptions increase API complexity but enhance reliability when used appropriately.
What are the potential problems with the following exception-based loop implementation, and why is it especially dangerous for debugging?
try { Iterator<Foo> i = collection.iterator(); while(true) { Foo foo = i.next(); // process foo } } catch (NoSuchElementException e) { }
Problems with this exception-based loop:
Performance degradation:
Readability issues:
Dangerous debugging problem:
Fails to utilize the Iterator API properly:
Correct approach:
for (Iterator<Foo> i = collection.iterator(); i.hasNext(); ) { Foo foo = i.next(); // process foo }
Or better:
for (Foo foo : collection) { // process foo }
What are runtime exceptions primarily intended to indicate, and how does this differ from checked exceptions? Give examples of appropriate uses for each.
Runtime Exceptions:
Examples of appropriate runtime exceptions:
NullPointerException
: Passing null to method requiring non-nullIllegalArgumentException
: Passing negative value to size parameterIndexOutOfBoundsException
: Accessing array beyond boundsUnsupportedOperationException
: Calling unimplemented optional methodClassCastException
: Incorrect type castingChecked Exceptions:
Examples of appropriate checked exceptions:
IOException
: File not found, but caller can try another locationSQLException
: Database connection lost, but can be retriedParseException
: Invalid date format, but user can correct inputFileNotFoundException
: Missing file that user could provideInsufficientFundsException
: Transaction that could succeed with less moneyThe key difference: Runtime exceptions typically indicate bugs that should be fixed in the code; checked exceptions indicate conditions that legitimate program flows need to handle.
What is exception translation and when should it be used?
Exception translation is when higher layers catch lower-level exceptions and throw exceptions that can be explained in terms of the higher-level abstraction.
Use it when:
Code pattern:
// Exception Translation try { // Use lower-level abstraction to do our bidding } catch (LowerLevelException e) { throw new HigherLevelException(...); }
This prevents breaking client code if implementation changes and maintains abstraction integrity.
Explain exception chaining with its implementation and benefits.
Exception chaining is a special form of exception translation where the lower-level exception is preserved as the "cause" of the higher-level exception.
Implementation:
// Exception Chaining try { // Use lower-level abstraction to do our bidding } catch (LowerLevelException cause) { throw new HigherLevelException(cause); } // Creating a chaining-aware exception class class HigherLevelException extends Exception { HigherLevelException(Throwable cause) { super(cause); } }
Benefits:
What are the best practices for documenting exceptions in Java APIs?
Best practices for documenting exceptions:
Document all exceptions, both checked and unchecked
Declare each checked exception individually in the throws clause
Document precisely when each exception is thrown
For interface methods, document unchecked exceptions thoroughly
For multiple methods throwing the same exception for the same reason:
Distinguish between checked and unchecked exceptions in documentation
Remember: Undocumented exceptions make it difficult or impossible for others to effectively use your classes and interfaces.
Compare and contrast the consequences of ignoring checked exceptions versus ignoring unchecked exceptions.
The advice against ignoring exceptions applies equally to both types, but with different implications:
Checked Exceptions:
Unchecked Exceptions:
Both cases result in a program that continues silently despite errors, potentially failing at arbitrary future points unrelated to the error source. The difference is primarily in what type of problem you're hiding.
Compare the empty catch block with propagating an exception. What are the debugging implications of each approach?
Empty Catch Block:
try { performOperation(); } catch (Exception e) { // Empty - silently continues }
Propagating Exception:
// Either no try-catch or re-throwing performOperation(); // or throw inside catch block
Even when you can't handle an exception properly, letting it propagate is vastly superior for debugging compared to silently ignoring it.
What design patterns or techniques can be used to implement failure atomicity in multi-step operations on complex objects?
Patterns for Failure Atomicity in Complex Operations:
Command Pattern: Encapsulate operations as objects that can track changes and undo them if needed
Memento Pattern: Capture and externalize object state before operations to restore if needed
Transactional Memory: Use software or hardware transactional memory systems that automatically roll back on failure
Copy-on-write: Create a complete copy before modification and only switch references on success
Builder Pattern with final commit: Build up changes in a separate object and apply them atomically at the end
Two-phase commit: Prepare all changes first (can be rolled back), then commit them if all preparations succeed
Snapshot isolation: Work on a logical snapshot of data, only updating the main state atomically on completion
The appropriate technique depends on performance requirements, object complexity, and the nature of the operations being performed.
Showing 10 of 34 cards. Add this deck to your collection to see all cards.