XPosed:挂钩在枚举中覆盖的函数

XPosed: Hooking a function overridden in an enum

假设我有一个这样定义的枚举 class(改编自 java documentation

package com.example.planetExample;

public enum Planet {
    MERCURY (3.303e+23, 2.4397e6){
        public double surfaceGravity() {
            return 42;
        }
    },
    VENUS   (4.869e+24, 6.0518e6);

    private final double mass;   // in kilograms
    private final double radius; // in meters
    Planet(double mass, double radius) {
        this.mass = mass;
        this.radius = radius;
    }

    // universal gravitational constant  (m3 kg-1 s-2)
    public static final double G = 6.67300E-11;

    double surfaceGravity() {
        return G * mass / (radius * radius);
    }
}

现在,我想使用 XPosed 挂钩 surfaceGravity 函数,该函数已被 MERCURY 覆盖(并且 而不是 下面定义的通用函数)。我怎样才能访问该功能?

我尝试了 findAndHookMethod("com.example.planetExample.Planet", lpparam.classLoader, "surfaceGravity", [etc]),但是那个只钩住了行星 class 定义的一般表面重力,而不是 MERCURY 定义的那个。如果我尝试 com.example.planetExample.Planet$MERCURYcom.example.planetExample.Planet.MERCURY,我会从 XPosed 收到错误消息,指出找不到函数 surfaceGravity

有没有办法使用 XPosed 挂钩这个函数?

MERCURY 是 Planet Enum 的一个字段。由于它有自己的实现,因此将在编译时为它生成一个 class,不幸的是它的名称与字段的名称不匹配(例如,在您的情况下它可能是 com.test.Planet$1)。

考虑以下示例:

public static void main(String[] args) {

    System.out.println("Mercury radius: " + Planet.MERCURY.surfaceGravity()); // 42
    System.out.println("Planet class: " + Planet.class.getName()); //prints "com.test.Planet"

    try {
        Class<?> planet_cls = Class.forName("com.test.Planet");

        System.out.println(Planet.class); // com.test.Planet
        System.out.println(Planet.MERCURY.getClass()); // com.test.Planet
        System.out.println(Planet.VENUS.getClass()); // com.test.Planet

        for(Class c: Planet.class.getDeclaredClasses())
            System.out.println("Name:" + c.getName()); // wont print

        for(Field c: Planet.class.getDeclaredFields())
            System.out.println("Field Name:" + c.getName()); // MERCURY & VENUS :)

        try {
            Field mercury_field = planet_cls.getDeclaredField("MERCURY");

            Object o = mercury_field.get(null);

            System.out.println("Field class name: " + o.getClass()); // com.test.Planet

            try {
                Method surfaceGravity = o.getClass().getDeclaredMethod("surfaceGravity");

                System.out.println("Confirm result: " + surfaceGravity.invoke(o)); // 42!

            } catch (NoSuchMethodException e) {
                e.printStackTrace();
            } catch (InvocationTargetException e) {
                e.printStackTrace();
            }

        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
}

因此通过按名称检索字段,您可以获得它的 class 和方法。请注意,在 Xposed 中,您将有另一个挂钩 api 接收方法而不是通过名称查找它,只需将示例中的方法传递给它即可。

该代码的输出可以节省您一些时间:

Mercury radius: 42.0
Planet class: com.test.Planet
class com.test.Planet
class com.test.Planet
class com.test.Planet
Field Name:MERCURY
Field Name:VENUS
Field Name:mass
Field Name:radius
Field Name:G
Field Name:$VALUES
Field class name: class com.test.Planet
Confirm result: 42.0