特殊的异常InvocationTargetException

InvocationTargetException是什么

JavaDoc一上来就说了:

InvocationTargetException is a checked exception that wraps an exception thrown by an invoked method or constructor.
InvocationTargetException是一个受检查异常,当被调用的方法或构造器内部抛出异常,该异常将会被包装成InvocationTargetException来进行接收

以下是知乎作者RednaxelaFX对其的描述

链接:https://www.zhihu.com/question/56718746/answer/150173531
InvocationTargetException 存在的意义是什么?
因为“原来的异常”无法直接以一种统一而又明确的方式表达出来,所以使用InvocationTargetException来将原来的异常包装起来,通过多加一层间接层的方式来提供统一的访问途径。
Java方法可以静态声明它可能会抛出一组固定的异常类型。而反射API里,Method.invoke() 与 Constructor.newInstance() 这些方法有“双重身份”——它们既代表要调用指定的目标方法,自身也是一个方法;目标方法可能会抛出异常,而它们自身在调用目标方法前也可能会抛出一些异常(例如IllegalArgumentException)。它们要调用的目标方法可能抛出任意Throwable类的派生类的异常,但它们自身却不能根据要调用的目标而“动态”改变自己声明要抛出的异常类型,而只能静态声明一组可能抛出的异常类型。声明抛出Throwable或Exception的话,这就太宽泛,难以准确反映异常的原因和意图;但不声明成这么宽泛的异常类型的话又无法完整覆盖所有可能由目标方法抛出的异常。那怎么办?简单,新增一个check exception类型,InvocationTargetException,将原本由目标方法抛出的异常包装起来,这样就可以给Method.invoke() / Constructor.newInstance() 的调用者一个统一的接口,既明确了“这个异常是由目标方法抛出的,不是由我自己抛出的”的意图,又能完整覆盖目标方法所能抛出的所有异常类型(InvocationTargetException.getTargetException() / getCause() 的类型是Throwable)。

大白话:在反射API里,Method.invoke() 与 Constructor.newInstance() 这些方法本身执行时会出现异常,同时被执行的方法或构造器内部也会出现异常,所以InvocationTargetException将被执行的方法或构造器内部抛出的异常进行包装,可通过InvocationTargetException.getTargetException() / getCause()进行获取,但不能直接使用InvocationTargetException,此时如果直接获取InvocationTargetException.getMessage()将会出现Null

直接获取被执行的方法或构造器内部抛出的异常的两种方式

  • JDK反射,将会将异常信息封装到InvocationTargetException,可通过InvocationTargetException.getTargetException() / getCause()进行获取
  • ASM反射,可直接通过Exception获取

Demo验证

  • 引入ASM相关依赖
<dependency>
    <groupId>com.esotericsoftware</groupId>
	<artifactId>reflectasm</artifactId>
	<!-- 版本自行修改 -->
	<version>1.11.7</version>
</dependency>

该依赖包底层使用的是如下依赖

<dependency>
	<groupId>org.ow2.asm</groupId>
	<artifactId>asm</artifactId>
	<version>5.1</version>
</dependency>
  • 测试用例如下:
import com.esotericsoftware.reflectasm.MethodAccess;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Objects;

public class AA {
    public static void main(String[] args) {

        Demo demo = new Demo();
        try {
            Method method = findMethod(Demo.class, "haha", new Object[]{"niuwa"});
            method.setAccessible(true);
            method.invoke(demo, 0);
        } catch (InvocationTargetException e) {
            // 如果直接使用InvocationTargetException来接收,则可以使用e.getTargetException() 来进行获取
            System.out.println("---------- JDK Invoke Exception ----------");
            // JDK 反射调用,将调用方法抛出的异常放置在Cause中了,Exception为空
            System.out.println("Cause message by JDK reflection:" + e.getTargetException().getMessage());
            System.out.println("Exception message by JDK reflection:" + e.getMessage());
        } catch (Exception e) {
            // 如果直接使用Exception来接收,则可以使用e.getCause() 来进行获取
            System.out.println("---------- JDK Invoke Exception ----------");
            // JDK 反射调用,将调用方法抛出的异常放置在Cause中了,Exception为空
            System.out.println("Cause message by JDK reflection:" + e.getCause().getMessage());
            System.out.println("Exception message by JDK reflection:" + e.getMessage());
        }

        try {
            MethodAccess access = MethodAccess.get(Demo.class);
            int setNameIndex = access.getIndex("haha");
            access.invoke(demo, setNameIndex, 0);
        } catch (Exception e) {
            System.out.println("---------- ASM Invoke Exception ----------");
            // JDK 反射调用,将调用方法抛出的异常直接放置在Exception中了,Cause为空
            System.out.println("Cause message by ASM reflection:" + e.getCause());
            System.out.println("Exception message by ASM reflection:" + e.getMessage());
        }
    }

    private static Method findMethod(Class<? extends Object> clazz, String methodName, Object[] args) {
        Method[] var6;
        int var5 = (var6 = clazz.getDeclaredMethods()).length;

        for (int var4 = 0; var4 < var5; ++var4) {
            Method method = var6[var4];
            if (method.getName().equals(methodName) && Objects.equals(method.getParameterTypes().length, args.length)) {
                return method;
            }
        }

        Class<?> superClass = clazz.getSuperclass();
        if (superClass != null) {
            return findMethod(superClass, methodName, args);
        } else {
            return null;
        }
    }
}

class Demo {
    public Demo() {

    }

    public void haha(int param) {
        System.err.println("计算结果:" + 10 / param);
    }
}
  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
`InvocationTargetException` 是 Java 反射机制中常见的异常,当调用某个方法或构造函数时,如果该方法或构造函数抛出了异常,那么反射机制会将原始的异常包装成 `InvocationTargetException` 异常并抛出。因此,我们在使用反射机制调用方法或构造函数时,需要特别注意该异常的处理。 在处理 `InvocationTargetException` 异常时,我们需要先通过 `getCause()` 方法获取原始异常,并根据原始异常的类型进行相应的处理。下面是一个示例代码: ```java try { Method method = MyClass.class.getMethod("myMethod"); method.invoke(new MyClass()); } catch (NoSuchMethodException | IllegalAccessException | InvocationTargetException e) { if (e instanceof InvocationTargetException) { Throwable cause = e.getCause(); if (cause instanceof MyException) { // 处理 MyException 异常 } else { // 处理其他异常 } } else { // 处理其他异常 } } ``` 在上述代码中,我们首先通过 `getMethod()` 方法获取指定类的指定方法,在使用 `invoke()` 方法调用该方法时,如果该方法抛出了异常,那么就会抛出 `InvocationTargetException` 异常。在捕获该异常后,我们先通过 `getCause()` 方法获取原始异常,并根据原始异常的类型进行相应的处理。 需要注意的是,`InvocationTargetException` 异常一般是由于被调用方法或构造函数内部抛出了异常导致的,因此我们在处理该异常时,通常要先检查原始异常的类型,并根据具体情况进行处理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

冰式的美式

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值