java.lang.reflect.invocationtargetexception – How to handle Invocation Target Exception
Reflection is commonly used by programs which require the ability to examine or modify the runtime behavior of applications running in the Java Virtual Machine. The reflection layer wraps any thrown exception as an InvocationTargetException
. In this way, it is clear whether the exception was actually caused by a failure in the reflection call, or a failure within the method called.
The InvocationTargetException
is a checked exception that wraps an exception thrown by an invoked method or constructor. The thrown exception is provided at construction time and can be accessed via the getTargetException
method. That exception is known as the cause
and can be accessed via the getCause
method.
For more information about the reflection in Java, please refer to the page here
.
Error case
The following code snippet throws an InvocationTargetException
:
ReflectionExample.java:
package main.java; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; public class ReflectionExample { @SuppressWarnings("unused") private int testMethod(String str) { if(str.length() == 0) throw new IllegalArgumentException("The string must contain at least one character!"); System.out.println("Inside testMethod: argument's value equals to: \"" + str + "\""); return 0; } public static void main(String... args) { try { // Retrieve an instance of the current class as an Object. Class<?> c = Class.forName("main.java.ReflectionExample"); Object t = c.newInstance(); Method[] declaredMethods = c.getDeclaredMethods(); for (Method method : declaredMethods) { String methodName = method.getName(); // Skip the current main method. if(methodName.contains("main")) continue; System.out.format("Invoking %s()%n", methodName); try { // Declare the method as accessible. method.setAccessible(true); /* Invoke the method with a 'null' parameter value, in order * for an exception to be thrown. */ Object returnValue = method.invoke(t, ""); System.out.format("%s() returned: %d%n", methodName, returnValue); } catch (InvocationTargetException ex) { System.err.println("An InvocationTargetException was caught!"); Throwable cause = ex.getCause(); System.out.format("Invocation of %s failed because of: %s%n", methodName, cause.getMessage()); } } } catch (ClassNotFoundException | InstantiationException | IllegalAccessException ex) { System.err.println("The following exception was thrown:"); ex.printStackTrace(); } } }
The result of the above snippet is:
Invoking testMethod() An InvocationTargetException was caught! Invocation of testMethod failed because of: The string must contain at least one character!
If we carefully observe the code, we will understand why the InvocationTargetException
was thrown. Initially, we get an instance of the ReflectionExample
class. Then, we iterate over its declared methods and we call the method under the name testMethod
, passing an empty String as an argument.
However, the testMethod
throws an IllegalArgumentException
, in case the length of the string equals to zero. That exception is wrapped as an InvocationTargetException
and is thrown in our sample application.
If we change the 39th line to:
Object returnValue = method.invoke(t, "Hello from Java Code Geeks!");
the execution continues without any exception being thrown. As a result, we get the following result:
Invoking testMethod() Inside testMethod: argument's value equals to: "Hello from Java Code Geeks!" testMethod() returned: 0
How to deal with the exception
First of all, coding an application using reflection is hard. A developer must have a strong grasp of the internal structure of the Java programming language, because the usage of reflection contains drawbacks and dangers, such as performance overhead and exposure of internal fields and methods.
If you decide to use reflection, consider enclosing your code inside a try-catch statement and manipulate the InvocationTargetException
accordingly. Notice that the result of the getCause
method can be one of the following:
- A null value.
- An unchecked exception, such as
RuntimeException
,IllegalArgumentException
,NullPointerException
, etc. - A checked exception, such as
NameNotFoundException
, etc. - A
java.lang.Error
, such asStackOverflowError
,OutOfMemoryError
, etc.
In your application’s code, make sure that you check for all aforementioned cases, otherwise your code may produce undesired bugs.
This was a tutorial about Java’s InvocationTargetException
.