These series are the reflection and study when I was leading an engineering team for all the aspects of software development. In particular, this post covers how to handle exception cases in a way that is safe from bugs.
In your career time, you’ve probably already seen various exceptions in different scenarios. For example, in Java, such as ArrayIndexOutOfBoundsException, NullPointerException. These exceptions generally indicate our software run into a state that could cause problem, and the and the information displayed by the language when the exception is thrown can help you find and fix the problem.
Exceptions for special results
Exceptions can be used to improve the structure of code that involves procedures with special results.
An unfortunately common way to handle special results is to return special values.
For example, lookup operations in the Java library are often designed like this: you get an index of -1 when expecting a positive integer, or a null reference when expecting an object. This approach is OK if used sparingly, but it has two problems.
- First, it’s tedious to check the return value.
- Second, it’s easy to forget to do it. (We’ll see that by using exceptions you can get help from the compiler in this.)
Also, it’s not always easy to find a ‘special value’. Suppose we have a Facebook class with a lookup method. Here’s one possible method signature:
final class Facebook {
func lookup(name: String) -> Person?
}
What should the method do if the facebook doesn’t have an entry for the person whose name is given? Well, we could return some special date that is not going to be used as a real date. Bad programmers have been doing this for decades; they would return 9/9/1970.
Or in Swift, they could return optional to indicate the a person with particular name could not be found. But its not obvious and robust enough, what if there are multiple errors happening during the lookup ? They will would not be able to identify which error it might occur, all we have is the optional now…
A better approach
The function throws an exception:
final class Facebook {
func lookup(name: String) throws -> Person? {
...
if ...not found...
throw LookupException.notFound(name: name)
...
}
}
and the caller handles the exception with a catch clause. For example:
let book: Facebook = ...
try {
let person = book.lookup("Alex")
} catch (LookupException.notFound) {
// his name was not in the face book
}
Now there’s no need for any special value, nor the checking associated with it.
Exception design considerations
The rule we have given — use exceptions for special results (i.e., anticipated situations), and to signal bugs (unexpected failures)— makes sense, but it isn’t the end of the story.
If you design a method to have its own (new) exception, you have to create a new class for the exception.
If you call a method that can throw a exception, you have to wrap it in a try - catch
statement (even if you know the exception will never be thrown).
This latter stipulation creates a dilemma.
Suppose, for example, you’re designing a queue abstraction.
Should popping the queue throw exception when the queue is empty?
Suppose you want to support a style of programming in the caller in which the queue is popped until the exception is thrown. So you choose to throw exception.
Now some caller wants to use the method in a context in which, immediately prior to popping,
the caller tests whether the queue is empty and only pops if it isn’t. Maddeningly, that caller will still need to wrap the call in a try - catch
statement.
This suggests a more refined rule:
- You should use an exception only to signal an unexpected failure (i.e. a bug), or if you expect that caller will usually write code that ensures the exception will not happen, because there is a convenient and inexpensive way to avoid the exception;
- Otherwise you should use a exception.
Here are some examples of applying this rule to hypothetical methods:
Queue.pop()
should not throws anEmptyQueueException
when the queue is empty, because it’s reasonable to expect the caller to avoid this with a call likeQueue.size()
orQueue.isEmpty()
.Url.getWebPage()
throws aIOException
when it can’t retrieve the web page, because it’s not easy for the caller to prevent this.integerSquareRoot(int x) -> Int
should throwsNotPerfectSquareException
when x has no integral square root, because testing whether x is a perfect square is just as hard as finding the actual square root, so it’s not reasonable to expect the caller to prevent it.