java 8里 Method方法bug

在写rpc框架的时候出现了头疼问题:

public Object invoke(Object bean)throws Exception {
        Method[] methods = clazz.getMethods();
        for (Method method:methods) {
            if(method.getName().equals(method.getName())){
                return method.invoke(bean, param);
            }
        }
        throw new Exception("找不到方法");
    }

这样写能找到方法,没有问题!!但是换成下面的方式就出问题了。

public Object invoke(Object bean)throws Exception {
        return clazz.getMethod(method, param.getClass()).invoke(bean, param);
    }

跑出异常为:
java.lang.NoSuchMethodException:com.robin.interf.UserService.getUser(java.lang.Integer)

java使用的版本是:jdk1.8.0_101

Class.java代码跟踪:
图片描述

是一个searchMethods的方法在查找,继续走
图片描述

代码走到这里,res=null,导致NoSuchMethodException异常,算是java8 bug吧!

原因:searchMethods方法里,m.getName() == internedName这句导致的,正确的是m.getName().intern() == internedName,因为m.getName()是堆里取出值,而internedName是常量池里的,而m.getName().intern()会将常量池里存在的字符串直接取到,不存在的会放入到常量池里。

建议:建议使用class.getMethods(),然后我们自己遍历,不推荐使用class.getMethod(methodName, paras)

阅读 4.3k
3 个回答

你的逻辑只匹配了方法名称并没有匹配参数类型吧。

public Object invoke(Object bean)throws Exception {
    Method[] methods = clazz.getMethods();
    for (Method method:methods) {
        if(method.getName().equals(method.getName())){
            return method.invoke(bean, param);
        }
    }
    throw new Exception("找不到方法");
}

上面的代码是楼主查询的method的方式吧,我说未匹配类型,是指楼主自己写的代码中没有匹配参数的类型。而jdk内部是有这个判断的。

public class Test {

    public String hello(int i) {
        return "Hello" + i;
    }

    public static void main(String[] args) {
        Test t = new Test();
        try {
            Method m = Test.class.getMethod("hello", Integer.class);
            System.out.println(m.invoke(t, 9));
        } catch (Exception e) {
            e.printStackTrace();
        }

        try {
            Method m = Test.class.getMethod("hello", int.class);
            System.out.println(m.invoke(t, 8));
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

我不太清楚楼主的代码是如何编写的,不过这段代码楼主可以运行一下看看区别。

public static void main(String[] args) throws InvocationTargetException {
    try {
        Class<?> c = Class.forName("java.lang.StringBuilder");
        Object instance = c.newInstance();
        Method m = c.getMethod("append", String.class);
        Object o = m.invoke(instance, "Hello World");
        System.out.println(o);
    } catch (ClassNotFoundException | NoSuchMethodException | IllegalAccessException | InstantiationException e) {
        e.printStackTrace();
    }
}

我也是jdk1.8.0_101, 完全没问题啊
新手上路,请多包涵

The parameterTypes parameter is an array of Class objects that identify the method's formal parameter types, in declared order. If parameterTypes is null, it is treated as if it were an empty array.
api里面说了,如果第二个参数不传,会默认空数组,也就是找入参为空的方法,那肯定会报‘找不到方法’的错了

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
宣传栏