2017-02-08 19 views
0

Скажем, у меня есть класс перечислимую, определенный как это (взято из java documentation)Экспоузд: Hooking функция переопределена в перечислении

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); 
    } 
} 

Теперь я хочу использовать Экспоузд крючок функцию surfaceGravity, которая была перекрываться MERCURY (и не общий, определенный ниже). Как я могу получить доступ к этой функции?

Я пробовал findAndHookMethod("com.example.planetExample.Planet", lpparam.classLoader, "surfaceGravity", [etc]), но это только перехватывает общую поверхностьGravity, определяемую классом Planet, а не тот, который определяется MERCURY. Если я попробую com.example.planetExample.Planet$MERCURY или com.example.planetExample.Planet.MERCURY, я получаю ошибки от XPosed, что функция surfaceGravity не может быть найдена.

Есть ли способ подключить эту функцию с помощью XPosed?

ответ

1

MERCURY - это поле Планеты Enum. Поскольку у него есть своя реализация, класс будет сгенерирован во время компиляции для него, к сожалению, его имя не будет соответствовать имени поля (например, в вашем случае это, скорее всего, будет 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$1 
     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$1 

      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(); 
    } 
} 

Так, извлекая поле по имени, вы можете получить его класс и методы. Обратите внимание, что в Xposed у вас будет еще один hooking api, который получает метод вместо поиска по имени, просто передайте ему метод в примере.

Вывод этого кода, чтобы сэкономить ваше время:

Mercury radius: 42.0 
Planet class: com.test.Planet 
class com.test.Planet 
class com.test.Planet$1 
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$1 
Confirm result: 42.0