先说结论:用 Ja va 反射机制,当然能从子类身上拿到父类定义的抽象方法元数据。关键在于搞清楚两个维度:一是方法的可见性(public 还是 protected/package-private),二是这个抽象方法究竟是在哪一层父类里第一次被声明的。搞明白了这两点,剩下的就是调哪几个 API 的问题。

从技术上说,核心操作就两步:拿到 Class 对象,然后调用反射 API。抽象方法只要被子类继承了(哪怕子类压根没重写),就一定能通过 getMethods() 或者递归扫描父类声明的方式拿到。具体的做法,咱们分几种情况来看。
先吃低垂的果实:拿所有 public 抽象方法
getMethods() 是个好东西,它返回的是当前类以及所有父类、接口里暴露出来的 public 方法——当然也包括抽象方法。但问题在于它不告诉你“这方法到底是从哪个类来的”,也不告诉你这方法到底是不是抽象的。所以得自己动手判断:
- 先调
clazz.getMethods(),拿到全部 public 方法数组 - 对每个方法,用
m.getModifiers()检查Modifier.isAbstract(m.getModifiers()) - 再用
m.getDeclaringClass() != clazz确认这方法是声明在父类(或接口)里的,不是子类自己定义的 - 一个小坑:接口里的 abstract 方法也会被混进来。如果你只关心类继承链上来的,别忘了加一层过滤——
!m.getDeclaringClass().isInterface()
这套组合拳下来,所有从父类继承过来的 public 抽象方法基本就稳了。
更细的活儿:拿到 protected 和 package-private 的抽象方法
不过,getMethods() 只能看见 public 的方法。假如父类里定义了一个 protected abstract void doWork();,那这个方法就不在 getMethods() 的返回里了。想拿它,就得自己把继承链捋一遍。
具体操作也不复杂:从 clazz.getSuperclass() 开始,向上递归遍历每一个父类,对每个父类调 getDeclaredMethods()。这个方法能取到该类所有可见性级别的方法——包括 public、protected 和 package-private。然后筛选出 Modifier.isAbstract(mod) 并且 !Modifier.isPublic(mod) 的那些即可。注意在碰到 Object.class 或者 null 的时候停下来,别死循环了。
搞清楚来源:谁是它的原始定义者?
这里有个容易混淆的点:getDeclaringClass() 指向的是这个方法第一次被声明的位置,而不一定是直接父类。举个例子:A 继承 B,B 继承 C,抽象方法定义在 C 里。那么对 A 的实例来说,这个抽象方法 getDeclaringClass() 返回的是 C.class。这其实挺合理的,但也意味着如果你想确认某个中间父类有没有重写这个方法(从而让它不再是抽象的),就得额外做一步检查——对那个父类调 getDeclaredMethod(m.getName(), m.getParameterTypes()),然后看它的修饰符。
顺便说一句:泛型和注解也没落下
抽象方法的参数类型、返回类型、异常列表和泛型结构,反射 API 全都给。想拿拿泛型,就上 getGenericReturnType()、getGenericParameterTypes()、getGenericExceptionTypes() 这一组方法;想看看方法上有没有 @Override 或自定义注解,就用 getAnnotation(...) 或者 getDeclaredAnnotations()。基本没有漏网之鱼。
