Exceptions, Assertions, Logging
-
What we’ll cover
Throwable Class
Basic Exception Handling
Better Exception Handling
Finally Keyword
Assertions
Logging
-
….a brief introduction:
Programs can fail for various reasons:
- Attempting to connect to an offline API or site
- Accessing an invalid index in an array
- Passing an unsupported value or incorrect object type to a method
Some of these issues are coding errors, while others are beyond the coder’s control. However, anticipating such errors and handling them appropriately is crucial.
-
Throwable Class
- Throwable Hierarchy
- Errors
- Checked Exceptions
- Unchecked Exceptions
-
Throwable Hierarchy
-
Errors
The Error
hierarchy describes internal errors and resource exhaustion situations.
- Do not advertise internal java errors; Any code can potentially throw an
Error
. IOError
,StackOverflowError
, andOutOfMemoryError
are a few of the commonly encounteredErrors
IOError
- serious I/O error has occurredStackOverflowError
- application recurses too deeply.OutOfMemoryError
- JVM cannot allocate an object because it is out of memory
-
Unchecked Exceptions
- Any exception which derives from
Error
orRuntimeException
class. - An Exception subclassing
RuntimeException
is considered to be the programmer’s fault.ArrayIndexOutOfBoundException
can be avoided by testing array index against array boundsNullPointerException
can be avoided by testing for null values.
-
Checked Exceptions
- An Exception subclassing
IOException
is potentially not the programmer’s fault.FileNotFoundException
can be thrown when trying to read from a remote file that a person incidentally removes.SQLException
can be thrown as a result of a faulty network connection.
-
Throwing Exceptions
Here are a couple of ways an exception can be thrown:
-
With a coding mistake:
String[] myEmptyArray = new String[0]; System.out.println(myEmptyArray[0]);
(throws an ArrayIndexOutOfBoundsException, since there is no index 0 in this array)
-
By throwing one explicitly, using the throw keyword:
throw new Exception(); throw new Exception("Oops! Something broke."); throw new RuntimeException(); throw new RuntimeException("Oops! I broke something.");
-
Basic Exception Handling
- What is Exception Handling?
- Unhandled Exception Example
- Handled Exception Example
-
What is Exception Handling?
- For exceptional situations, Java uses a form of error trapping called, exception handling.
- Exception handling enables the author of the code to record and handle errors in their program.
-
Unchecked Unhandled Exception Example; Compile Error
import java.io.*;
class FilePrinter {
private final BufferedReader reader;
public FilePrinter(String fileDirectory) {
// What if the file does not exist?
this.reader = new BufferedReader(new FileReader(fileDirectory));
}
public void printFile() {
String line = null;
do {
// What if the System fails to read in the next line?
// (For example if the file was suddenly closed, modified, or deleted)
line = reader.readLine();
System.out.println(line);
} while (line != null);
}
}
-
Exception Handling; Signature Throw Clause
import java.io.*;
class FilePrinter {
private final BufferedReader reader;
public FilePrinter(String fileDirectory) throws FileNotFoundException {
this.reader = new BufferedReader(new FileReader(fileDirectory));
}
public void printFile() throws IOException {
String line = null;
do {
line = reader.readLine();
System.out.println(line);
} while (line != null);
}
}
-
Exception Handling; Try / Catch
import java.io.*;
class FilePrinter {
private final BufferedReader reader;
public FilePrinter(String fileDirectory) throws FileNotFoundException {
this.reader = new BufferedReader(new FileReader(fileDirectory));
}
public void printFile() throws IOException {
String line = null;
do {
line = reader.readLine();
System.out.println(line);
} while (line != null);
}
public void tryPrintFile() {
try {
printFile();
} catch (IOException ioe) {
ioe.printStackTrace();
}
}
}
-
Better Exception Handling
- Multi-Exception Handling
- Dynamic Exception Handling
- Uniform Exception Handling (Good)
- Uniform Exception Handling (Bad)
- How to throw an Exception
- Recursion
andException Handling
-
Multi-Exception Handling
- Consider the case where multiple exceptions may be thrown.
- For example, in our
FilePrinter
class, the- constructor throws a
FileNotFoundException
printFile()
throws anIOException
- constructor throws a
- What if we wanted to create a
FilePrinter
object, then print its contents?
-
Multi-Exception Handling Examples
public class FilePrinterTest {
private static final String invalidFileName = "";
@Test(expected = FileNotFoundException.class)
public void testInstantiation() throws FileNotFoundException {
FilePrinter fpt = new FilePrinter(invalidFileName);
}
// Attempt to instantiate FilePrinter with invalid name
// Attempt to invoke method on unininstatiated FilePrinter object
@Test(expected = NullPointerException.class)
public void testNullPointer() throws NullPointerException {
FilePrinter fpt = null;
try {
fpt = new FilePrinter(invalidFileName);
} catch (FileNotFoundException e) {
System.out.println("Printing stack trace...");
e.printStackTrace();
}
fpt.tryPrintFile();
}
@Test(expected = NullPointerException.class)
public void testMultiThrowSignature() throws NullPointerException, FileNotFoundException {
testNullPointer();
testInstantiation();
}
}
-
Dynamic Exception Handling; Expanded
public class FilePrinterTest {
private static final String invalidFileName = "";
public void testInstantiateAndPrint() {
FilePrinter fpt = null;
try {
fpt = new FilePrinter(invalidFileName);
} catch(FileNotFoundException fnfe) {
fnfe.printStackTrace();
}
try {
fpt.printFile();
} catch (IOException e) {
e.printStackTrace();
}
}
}
-
Dynamic Exception Handling; Compressed
public class FilePrinterTest {
private static final String invalidFileName = "";
public void testInstantiateAndPrint() {
FilePrinter fpt = null;
try {
fpt = new FilePrinter(invalidFileName);
fpt.printFile();
} catch(FileNotFoundException fnfe) {
// handle FileNotFoundException
fnfe.printStackTrace();
} catch(IOEXception ioe) {
// handle IOException
ioe.printStackTrace();
}
}
}
-
Uniform Handling Of Exceptions (Good)
public class FilePrinterTest {
private static final String invalidFileName = "";
public void testInstantiateAndPrint() {
FilePrinter fpt = null;
try {
fpt = new FilePrinter(invalidFileName);
fpt.printFile();
// bit-wise operator supported by java 1.7+
} catch(FileNotFoundException | IOException exception) {
// handle all exceptions the same way
exception.printStackTrace();
}
}
}
- Each expected exception in this class is explicitly named.
- The handling of each of them is uniform.
-
Uniform Handling Of Exceptions (Bad)
public class FilePrinterTest {
private static final String invalidFileName = "";
public void testInstantiateAndPrint() {
FilePrinter fpt = null;
try {
fpt = new FilePrinter(invalidFileName);
fpt.printFile();
} catch(Exception exception) {
// handle all exceptions the same way
exception.printStackTrace();
} catch(IllegalArgumentException iae) {
iae.printStackTrace();
}
}
public void parseIntegerInput(String s) {
try {
Long.parseLong(s);
} catch(NumberFormatException e) {
e.printStackTrace();
throw new IllegalArgumentException();
}
}
}
-
Recursion and Exception Handling
- DON’T DO IT!
- Recursion and Exception Handling do not go together
- Exceptions keep track of all pending method calls
- By nature, recursion pends method calls
n
levels deep, wheren
is the recursive depth of the method call. - Combining recursion and exception handling can result in very strange
StackTraces
-
Finally Keyword
- Purpose
- Conditions under which
finally
block is executed - Syntax
- Decoupling
finally
clause fromtry/catch
clauses
-
Purpose
- When code throws an exception, it stops processing the remaining code in the scope, then exits the method.
- If the method has acquired some local resource, then this can become an issue: The program will cease execution, and hold the resource indefinitely.
- The finally clause executes whether or not an exception was code.
-
Conditions under which finally
block is executed
- If no exception are thrown.
- If exception outside
try
block is thrown. - If an exception is thrown in a
catch
clause. - The program skips to the
finally
clause, if thecatch
clause does not throw an exception.
-
Decoupling finally
clause from try/catch
clauses
class BookExample {
public void example1() {
InputStream in = ... ;
try {
try {
// code that may throw exception
} finally {
in.close();
}
} catch(IOException ioe) {
/// handle exception some way
}
}
}
-
Exceptions and Inheritance
When a method overrides a method from a superclass or interface, it is not allowed to add checked exceptions. It is, however, allowed to declare fewer exceptions or declare a subclass of a declared exception. Methods declare exceptions with the keyword throws.
-
Exceptions and Inheritance (continued)
Adding a checked exception to a method won’t compile:
class HasNoMovesException extends Exception { }
class Dancer {
public void dance() { }
}
class Boogie extends Dancer {
public void dance() throws HasNoMovesException {
// DOES NOT COMPILE
}
}
-
Exceptions and Inheritance (continued)
…declaring fewer exceptions, or declaring a subclass of the exception type are perfectly legal, however.
class Dancer {
public void hop() throws HasNoMovesException { }
}
class Boogie extends Dancer {
public void dance() { }
}
class Dancer {
public void dance() throws Exception { }
}
class Boogie extends Dancer {
public void dance() throws HasNoMovesException { }
}
-
You could also throw it explicitly:
public void dance() {
throw new RuntimeException("I have no moves!");
}
-
Assertions
- Assertions are commonly used idiom of defensive programming.
- Java has a keyword
assert
, which takes two forms:assert condition;
assert condition : expression;
assert
evaluates a condition, then throws anAssertionError
if it is false. The second argument expression is a message String.
-
Toggling Assert Statements
- By default, assertions are disabled: If an assert statement is passed
false
, no exception is thrown. - Assertions can be enabled by running the program with the
-ea
option. java -ea MyProject
enables for entire projectjava -ea:MyClass -ea:com.zipcodewilmington.MyProject
enables forMyClass
-
When To Use
- Assertion failures are intended to be fatal, unrecoverable errors
- Assertion checks are turned on only during development and testing
- As an additional check against uncanny method returns.
-
Logging
- It’s common to use
System.out.println
to check against troublesome code. - Once the issue is resolved, these statements are usually removed, or commented out.
- Later, if the issue persists, the print statements are re-inserted.
- The Logging API is designed to overcome this issue.
-
Principal advantages of Logging API
- It’s easy to (un)suppress all log records, or just those below a certain level.
- Suppressed logs are inexpensive; The penalty for leaving them in your code is minimal.
- Log records can be directed to different handlers; Console display, writing to file / database, etc.
- Log records can be formatted; For example, plaint ext, or XML
- Logging configuration is controlled by configuration file; Applications can replace this mechanism
-
The 7 Logging Levels
- By default, loggers will log all messages of
INFO
or higher to console. SEVERE
WARNING
INFO
CONFIG
FINE
FINER
FINEST
-
Syntax
public class LogDemo {
// it is advised that you name your logger the same as your Main Application package
Logger logger = Logger.getLogger("com.zipcodewilmington.MainApplication");
public void logTest() {
logger.setLevel(Level.SEVERE); // log severe
logger.setLevel(Level.WARNING); // log severe, warning
logger.setLevel(Level.INFO); // log severe, warning, info
logger.setLevel(Level.CONFIG); // log severe, warning, info, config
logger.setLevel(Level.FINE); // log severe, warning, info, config, fine
logger.setLevel(Level.FINER); // log severe, warning, info, config, fine, finer
logger.setLevel(Level.FINEST); // log severe, warning, info, config, fine, finer, finest
}
}
-