为了账号安全,请及时绑定邮箱和手机立即绑定

java 8里 Method方法bug

java 8里 Method方法bug

慕田峪7331174 2019-03-13 18:15:43
在写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_101Class.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 回答

?
HUWWW

TA贡献1874条经验 获得超12个赞

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


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

        }

    }

}

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


查看完整回答
反对 回复 2019-04-25
?
慕桂英3389331

TA贡献2036条经验 获得超8个赞

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, 完全没问题啊


查看完整回答
反对 回复 2019-04-25
?
GCT1015

TA贡献1827条经验 获得超4个赞

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里面说了,如果第二个参数不传,会默认空数组,也就是找入参为空的方法,那肯定会报‘找不到方法’的错了

查看完整回答
反对 回复 2019-04-25
  • 4 回答
  • 0 关注
  • 552 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信