It is not a matter of IF but WHEN things will go wrong in a computer program. Sometimes there are bugs, errors of one form or another. There are also unforeseen use cases. You can never assume a computer program is perfect. Exception-Handling helps us to catch erroneous events and devise means of correcting them. We discuss this topic here since exception-handling can take more code than should be put into the main line of execution. In such cases, a method in an exception-handling class should be called. Exception handling mechanisms allow a program to continue executing, instead of terminating it abruptly, even if an error occurs in the program.
10.5 Error Handling and Robust Program Design
To Fix or Not to Fix
Let’s now consider a problem where it is less clear whether an exception can be successfully fixed “on the fly.” Suppose you have a program
that contains an array of Strings, which is initially created with just two
elements.
If an attempt is made to add more than two elements to the array, an
ArrayIndexOutOfBoundsException will be raised. This exception
can be handled by extending the size of the array and inserting the element. Then the program’s normal execution can be resumed.
To begin creating such a program, let’s first design a method that will
insert a string into the array. Suppose that this is intended to be a private method that will only be used within the program. Also, let’s suppose that
the program maintains a variable, count, that tracks how many values
have been stored in the array. Therefore, it will not be necessary to pass
the array as a parameter. So, we are creating a void method with one
parameter, the String to be inserted:
The comment notes where an exception might be thrown.
Can we handle this exception? When this exception is raised, we could
create a new array with one more element than the current array. We
could copy the old array into the new array and then insert the String in the new location. Finally, we could set the variable list, the array reference, so that it points to the new array. Thus, we could use the following
try/catch block to handle this exception:
The effect of the catch clause is to create a new array, still referred to as
list, but that contains one more element than the original array.
Note the use of the finally clause here. For this problem it’s important that we incrementally count in the finally clause. This is the only way to guarantee that count is incremented exactly once whenever an element is assigned to the array.
The design of the FixArrayBound class is shown in Figure 10.17. It provides a simple GUI interface that enables you to test the insertString() method. This program has a standard Swing interface, using a JFrame as the top-level window. The program’s components are contained within a JPanel that’s added to the JFrame in the main() method.
Each time the user types a string into the text field, the actionPerformed() method calls the insertString() method to add the
string to the array. On each user action, the JPanel is repainted. The
paintComponent() method simply clears the panel and then displays
the array’s elements (Fig. 10.18).
The complete implementation of FixArrayBound is given in Figure 10–19. This example illustrates how an exception can be handled successfully and the program’s normal flow of control resumed. However,
the question is whether such an exception should be handled this way.
Unfortunately, this is not a well-designed program. The array’s initial
size is much too small for the program’s intended use. Therefore, the fact
that these exceptions arise at all is the result of poor design. In general,
exceptions should not be used as a remedy for poor design.
For a program that uses an array, the size of the array should be chosen so that it can store all the objects required by the program. If the program is
some kind of failsafe program, which cannot afford to crash, then something like the previous approach might be justified, provided this type
of exception occurs very rarely. Even in that case it would be better to
generate a message that alerts the program’s user that this condition has
occurred. The alert will indicate a need to modify the program’s memory
requirements and restart the program.
Figure 10.18: The strings displayed are stored in an array that
is extended each time a new string
is entered.
If it is not known in advance how many objects will be stored in an
array, a better design would be to make use of the java.util.Vector class (see “From the Java Library” in Chapter 9). Vectors are designed to
grow as new objects are inserted. In some ways the exception-handling
code in our example mimics the behavior of a vector. However, the
Vector class makes use of efficient algorithms for extending its size. By
contrast, exception-handling code is very inefficient. Because exceptions
force the system into an abnormal mode of execution, it takes considerably
longer to handle an exception than it would to use a Vector for this type
of application.
SELF-STUDY EXERCISE
EXERCISE 10.13 For each of the following exceptions, determine whether it can be handled in such a way that the program can be resumed or whether the program should be terminated:
a. A computer game program detects a problem with one of its GUI elements and throws a NullPointerException.
b. A factory assembly-line control program determines that an important control value has become negative and generates an ArithmeticException.
c. A company’s Web-based order form detects that its user has entered an
invalid String and throws a SecurityException.