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$MERCURY
或 com.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
假设我有一个这样定义的枚举 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$MERCURY
或 com.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