你知道Java 8 的方法引用吗
1. 前言
Java中的方法引用,很多同学都见过但却叫不出名字甚至不太会用,在这篇文章中,我们将看到什么是方法引用以及如何使用它。
2. 方法引用的使用场景
我们先来看看方法引用的使用:
new Random().ints(10)
.map(i->Math.abs(i))
.forEach(i -> System.out.println(i));
这里我们随机生成 10 个整数然后取它们绝对值并一一打印出来。写法是没有问题的,但是还是可以再简化的。
map
方法接受的是一个函数式接口IntUnaryOperator
,那么上面代码中的i->Math.abs(i)
实际上是:
new IntUnaryOperator() {
@Override
public int applyAsInt(int operand) {
return Math.abs(operand);
}
}
从上面来看IntUnaryOperator
就是代理了Math.abs(int i)
,参数列表、返回值都相同,而且没有掺杂其它额外的逻辑。这一点非常重要,不掺杂其它逻辑才能相互代替。那么就可以通过方法引用来简化Lambda 表达式。上面的式子就可以简化为:
new Random().ints(10)
.map(Math::abs)
.forEach(System.out::println);
3. 方法引用
Java 方法引用是Java 8随着Lambda表达式引入的新特性。可以直接引用已有Java类或对象的方法或构造器。方法引用通常与Lambda表达式结合使用以简化代码。其使用条件是: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 表达式都可以看作一个执行策略,方法引用反而让你能更加清楚执行了什么策略。另外我经常见到类似如下的流式写法:
new Random().ints(10)
.map(operand -> {
System.out.println("operand = " + operand);
return operand+1;
})
.forEach(System.out::println);
这种"大肚子"写法的风格是不建议在函数式编程中出现的。最好单独提出来封装做方法引用,写成下面的风格:
public void randomInt() {
new Random().ints(10)
.map(this::selfIncreasing)
.forEach(System.out::println);
}
// 封装
private int selfIncreasing(int self){
System.out.println("self = " + self);
return self+1;
}
这样反而可读性很强,随机取 10 个数,然后每个数走个自增并分别打印出来。
5. 总结
方法引用实现在特定场景下Lambda 表达式的简化表示,目的在于让代码更加简洁。但是习惯了传统Java编程风格的同学上来会不太适应,希望借助于本文能帮助你解决这个问题。
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- Java——Stream数据流
- JavaWeb——MyBatis框架之连接池原理、MyBatis事务提交设置、动态SQL语法总结
- 8种ETL算法归纳总结,看完这些你就全明白了
- JavaWeb——Maven基础之详细总结,从零开始搭建Maven工程,包含一些常见的坑org.eclipse.jdt.internal.compiler.classfmt.ClassFormatEx
- JavaWeb——Redis数据库之Jedis操作5种类型数据的使用总结与前端Ajax获取Redis缓存数据的案例实战(结合了MySQL数据库)
- Java——数据库编程JDBC之快速入门吐血总结及各关键对象详解(提供了JDBCUtils工具类)
- MySQL数据库——事务的操作(开启、回滚、提交)、特征、隔离级别基础总结
- MySQL数据库——数据库的设计(多表之间的关系与三大范式)与备份还原
- JavaWeb——JSP入门学习(JSP基本概念、JSP脚本、JSP内置对象)
- MySQL数据库——数据库CRUD之基本DML增删改表操作及DQL查表操作
- JavaWeb——AJAX异步技术实现方式与案例实战(原生的JS方式、使用JQuery方式)
- JavaWeb——一文带你入门Servlet(生命周期、注解配置方法、IDEA与tomcat的相关配置)
- JavaWeb——JQuery之基础案例实战(实现表格隔行换色、实现全选全不选、QQ表情选择、下拉列表选中条目左右选择功能)
- JavaWeb——JQuery之DOM操作应用及实践案例总结(DOM内容操作、DOM属性操作、CRUD操作)
- JavaWeb——Filter过滤器快速入门与是否登录验证&过滤敏感词汇案例实战(Filter配置方式、执行流程、生命周期方法、过滤器链)