Java中的方法引用,很多同学都见过但却叫不出名字,在这篇文章中,我们将看到什么是方法引用以及如何使用它。
1. 前言
Java中的方法引用,很多同学都见过但却叫不出名字,在这篇文章中,我们将看到什么是方法引用以及如何使用它。
2. 方法引用的使用场景
我们先来看看方法引用的使用:
- newRandom().ints(10)
- .map(i->Math.abs(i))
- .forEach(i->System.out.println(i));
这里我们随机生成10个整数然后取它们绝对值并一一打印出来。写法是没有问题的,但是还是可以再简化的。
map方法接受的是一个函数式接口IntUnaryOperator,那么上面代码中的i->Math.abs(i)实际上是:
- newIntUnaryOperator(){
- @Override
- publicintapplyAsInt(intoperand){
- returnMath.abs(operand);
- }}
从上面来看IntUnaryOperator就是代理了Math.abs(int i),参数列表、返回值都相同,而且没有掺杂其它额外的逻辑。这一点非常重要,不掺杂其它逻辑才能相互代替。那么就可以通过方法引用来简化Lambda 表达式。上面的式子就可以简化为:
- newRandom().ints(10)
- .map(Math::abs)
- .forEach(System.out::println);
3. 方法引用
Java 方法引用是Java 8随着Lambda表达式引入的新特性。 可以直接引用已有Java类或对象的方法或构造器。方法引用通常与Lambda表达式结合使用以简化代码。其使用条件是:Lambda表达式的主体仅包含一个表达式,且Lambda表达式只调用了一个已经存在的方法;被引用的方法的参数列表和返回值与Lambda表达式的输入输出一致。
3.1 格式
方法引用的格式为<ClassName | instance>::<MethodName>。也就是被引用的方法所属的类名和方法名用双冒号::隔开,构造器方法是个例外,引用会用到new关键字,总结了一下:
引用方式说明静态方法引用ClassName :: staticMethodName 例如上面的Math::abs构造器引用ClassName :: new 例如通过Supplier<T> 返回新实例类任意实例方法引用ClassName :: instanceMethodName 例如 String::concat类特定实例方法引用instance:: instanceMethodName 例如 this::equals
4. 关于可读性问题
大部分人认为Lambda 表达式存在阅读困难的问题,其实不然,这种流水线的结构恰恰增加了可读性,每一个Lambda 表达式都可以看作一个执行策略,方法引用反而让你能更加清楚执行了什么策略。另外我经常见到类似如下的流式写法:
- newRandom().ints(10)
- .map(operand->{
- System.out.println("operand="+operand);
- returnoperand+1;
- }).forEach(System.out::println);
这种"大肚子"写法的风格是不建议在函数式编程中出现的。最好单独提出来封装做方法引用,写成下面的风格:
- publicvoidrandomInt(){
- newRandom().ints(10)
- .map(this::selfIncreasing)
- .forEach(System.out::println);
- }//封装
- privateintselfIncreasing(intself){
- System.out.println("self="+self);
- returnself+1;
- }
这样反而可读性很强,随机取10个数,然后每个数走个自增并分别打印出来。
5. 总结
方法引用实现在特定场景下Lambda表达式的简化表示,目的在于让代码更加简洁。但是习惯了传统Java编程风格的同学上来会不太适应,希望借助于本文能帮助你解决这个问题。
©本文为清一色官方代发,观点仅代表作者本人,与清一色无关。清一色对文中陈述、观点判断保持中立,不对所包含内容的准确性、可靠性或完整性提供任何明示或暗示的保证。本文不作为投资理财建议,请读者仅作参考,并请自行承担全部责任。文中部分文字/图片/视频/音频等来源于网络,如侵犯到著作权人的权利,请与我们联系(微信/QQ:1074760229)。转载请注明出处:清一色财经