Java Lambda 表达式的一个重要用法是简化某些匿名内部类(Anonymous Classes)的写法。实际上 Lambda 表达式并不仅仅是匿名内部类的语法糖,JVM 内部是通过 invokedynamic 指令来实现 Lambda 表达式的。具体原理放到下一篇。本篇我们首先感受一下使用 Lambda 表达式带来的便利之处。
取代某些匿名内部类
本节将介绍如何使用 Lambda 表达式简化匿名内部类的书写,但 Lambda 表达式并不能取代所有的匿名内部类,只能用来取代函数接口(Functional Interface)的简写。先别在乎细节,看几个例子再说。
例子 1:无参函数的简写
如果需要新建一个线程,一种常见的写法是这样:
上述代码给 Tread 类传递了一个匿名的 Runnable 对象,重载 Runnable 接口的 run()方法来实现相应逻辑。这是 JDK7 以及之前的常见写法。匿名内部类省去了为类起名字的烦恼,但还是不够简化,在 Java 8 中可以简化为如下形式:上述代码跟匿名内部类的作用是一样的,但比匿名内部类更进一步。这里连接口名和函数名都一同省掉了,写起来更加神清气爽。如果函数体有多行,可以用大括号括起来,就像这样:
例子 2:带参函数的简写
如果要给一个字符串列表通过自定义比较器,按照字符串长度进行排序,Java 7 的书写形式如下:
上述代码通过内部类重载了 Comparator 接口的 compare()方法,实现比较逻辑。采用 Lambda 表达式可简写如下:
上述代码跟匿名内部类的作用是一样的。除了省略了接口名和方法名,代码中把参数表的类型也省略了。这得益于 javac 的类型推断机制,编译器能够根据上下文信息推断出参数的类型,当然也有推断失败的时候,这时就需要手动指明参数类型了。注意,Java 是强类型语言,每个变量和对象都必需有明确的类型。
简写的依据
也许你已经想到了,能够使用 Lambda 的依据是必须有相应的函数接口(函数接口,是指内部只有一个抽象方法的接口)。这一点跟 Java 是强类型语言吻合,也就是说你并不能在代码的任何地方任性的写 Lambda 表达式。实际上 Lambda 的类型就是对应函数接口的类型。Lambda 表达式另一个依据是类型推断机制,在上下文信息足够的情况下,编译器可以推断出参数表的类型,而不需要显式指名。Lambda 表达更多合法的书写形式如下:
类型推断
上述代码中,1 展示了无参函数的简写;2 处展示了有参函数的简写,以及类型推断机制;3 是代码块的写法;4 和 5 再次展示了类型推断机制。自定义函数接口自定义函数接口很容易,只需要编写一个只有一个抽象方法的接口即可。
上面代码中的@FunctionalInterface 是可选的,但加上该标注编译器会帮你检查接口是否符合函数接口规范。就像加入@Override 标注会检查是否重载了函数一样。有了上述接口定义,就可以写出类似如下的代码:
- END -
往期推荐
图解分析:Kafka 生产者客户端工作原理
isEmpty 和 isBlank 区别?
Kafka 发送消息过程中拦截器的用途?
Java线程的join操作有什么作用?
Nginx 最常用的两个功能:负载均衡和缓存
领取专属 10元无门槛券
私享最新 技术干货