Android中的反思无效

时间:2011-11-08 13:15:52

标签: java android reflection

我尝试使用反射为目标api级别为7的应用程序使用自定义列表视图。必要的文件只能从api级别9获得,所以我尝试通过反射修复它。

我需要找到受保护的Method View.overScrollBy(int,int,int,int,int,int,int,int,boolean)。当我打电话

View.getDeclaredMethods() 

并遍历Method []数组我找到它,但是当我尝试

View.class.getDeclaredMethod(String name, Class...< ? > paramTypes)

我得到NoSuchMethodException。我将硬编码的Method Name和parameterType值与从方法中提取的值(通过迭代找到)进行了比较,它们是相同的......

private boolean initCompatibility() 
{
Method[] methods = View.class.getDeclaredMethods();
try {
    // The name of the Method i am looking for;
    String OVERSCROLL_S = "overScrollBy";
    for (Method meth : methods) {
        if (meth.getName().equals(OVERSCROLL_S)) {
            mListView_overScrollBy = meth;
            break;
            // method found
        }
    }

    // Params for getDeclaredMethod(…)
    String methodName = "overScrollBy";
    Class[] methodParams =  {  Integer.TYPE, Integer.TYPE, Integer.TYPE, 
            Integer.TYPE, Integer.TYPE, Integer.TYPE, Integer.TYPE, 
            Integer.TYPE, Boolean.TYPE };

    // works
    Method test =  View.class.getDeclaredMethod(methodName,methodParams);
    // fails
    View.class.getDeclaredMethod(mListView_overScrollBy.getName(), 
               mListView_overScrollBy.getParameterTypes());

    /*
    * I also tried this way around and again the first worked and the second
    * failed, so the input arguments are not the problem...
    * View.class.getDeclaredMethod( mListView_overScrollBy.getName(), 
    *                           mListView_overScrollBy.getParameterTypes() );
    * Method test =  View.class.getDeclaredMethod(methodName,methodParams);
    */

    return true;
    } catch (SecurityException e) {
        e.printStackTrace();
        return false;
    } catch (NoSuchMethodException e) {
        e.printStackTrace();
        return false;
    }
}

我不明白为什么电话总是第一次工作而第二次不工作。有趣的是,当我只为View.class.getDeclaredMethod(String name,Class ...&lt;?&gt; paramTypes)调用一次时,它也会失败,并且无论我使用的是硬编码输入值还是从中提取的值都没有任何区别我正在寻找的方法......

有人知道问题是什么吗?感谢

1 个答案:

答案 0 :(得分:1)

这非常有趣,但我认为它不是特定于Android的。

我用简单的Java编写了这个小测试:

    public class ReflectionTest {

        public static void main(String[] args){
            Method[] m = ReflectionTest.class.getDeclaredMethods();
            for (Method method : m) {
                System.out.println(method.getName());
            }               

            try {
                Method m1 = ReflectionTest.class.getDeclaredMethod("d0", int.class, boolean.class);
                if(m1 != null){
                    System.out.println("m1 found!");
                }
            } catch (Exception e) {
                e.printStackTrace();
            } 

            try {
                Method m2 = ReflectionTest.class.getDeclaredMethod("d0", Integer.TYPE, Boolean.TYPE);
                if(m2 != null){
                    System.out.println("m2 found!");
                }
            } catch (Exception e) {
                e.printStackTrace();
            } 

            try {
                Class<?>[] carr = m[1].getParameterTypes();
                Method m3 = ReflectionTest.class.getDeclaredMethod("d0", carr);
                if(m3 != null){
                    System.out.println("m3 found!");
                }
            } catch (Exception e){
                e.printStackTrace();
            } 
        }

        public void d0(int a, boolean b){

        }
    }

在eclipse中,如果我调试它,则打印三个m1,m2和m3。但是,如果我运行它,在尝试获取m3时会抛出NoSuchMethodException

更新

  • 在linux下使用jre 7测试运行,并打印了所有三个m1,m2和m3。也许是jre6的问题?或者是eclipse运行配置?
  • 将格式建议的carr声明更改为使用方法0而不是1:Class<?>[] carr = m[0].getParameterTypes();。现在它运行正常但在调试模式下抛出异常。这意味着返回的数组m的方法顺序不同。
  • 更新#2已确认,我已经包含了一个for循环来打印方法名称。在运行模式下,与调试模式相比,方法数组的顺序相反。