java 8里 Method方法bug
高洛峰
高洛峰 2016-10-27 10:24:34
0
1
538

在写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代码跟踪:

1.png

是一个searchMethods的方法在查找,继续走

1.png

代码走到这里,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)


高洛峰
高洛峰

拥有18年软件开发和IT教学经验。曾任多家上市公司技术总监、架构师、项目经理、高级软件工程师等职务。 网络人气名人讲师,...

全員に返信(1)
三叔

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

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

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

いいねを押す +0
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート
私たちについて 免責事項 Sitemap
PHP中国語ウェブサイト:福祉オンライン PHP トレーニング,PHP 学習者の迅速な成長を支援します!