Java8 Lambda表达式详解手册及实例
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/wo541075754/article/details/102530810
先贩卖一下焦虑,Java8发于2014年3月18日,距离现在已经快6年了,如果你对Java8的新特性还没有应用,甚至还一无所知,那你真得关注公众号“程序新视界”,好好系列的学习一下Java8的新特性。Lambda表达式已经在新框架中普通使用了,如果你对Lambda还一无所知,真得认真学习一下本篇文章了。
现在进入正题Java8的Lambda,首先看一下发音 ([ˈlæmdə])表达式。注意该词的发音,b是不发音的,da发[də]音。
为什么要引入Lambda表达式
简单的来说,引入Lambda就是为了简化代码,允许把函数作为一个方法的参数传递进方法中。如果有JavaScript的编程经验,马上会想到这不就是闭包吗。是的,Lambda表达式也可以称作Java中的闭包。
先回顾一下Java8以前,如果想把某个接口的实现类作为参数传递给一个方法会怎么做?要么创建一个类实现该接口,然后new出一个对象,在调用方法时传递进去,要么使用匿名类,可以精简一些代码。以创建一个线程并打印一行日志为例,使用匿名函数写法如下:
在java8以前,使用匿名函数已经算是很简洁的写法了,再来看看使用Lambda表达式,上面的代码会变成什么样子。
是不是简洁到爆!
我们都知道java是面向对象的编程语言,除了部分简单数据类型,万物皆对象。因此,在Java中定义函数或方法都离不开对象,也就意味着很难直接将方法或函数像参数一样传递,而Java8中的Lambda表达式的出现解决了这个问题。
Lambda表达式使得Java拥有了函数式编程的能力,但在Java中Lambda表达式是对象,它必须依附于一类特别的对象类型——函数式接口(functional interface),后面详细讲解。
Lambda表达式简介
Lambda表达式是一种匿名函数(对Java而言这并不完全准确),通俗的说,它是没有声明的方法,即没有访问修饰符、返回值声明和名字的方法。使用Lambda表达式的好处很明显就是可以使代码变的更加简洁紧凑。
Lambda表达式的使用场景与匿名类的使用场景几乎一致,都是在某个功能(方法)只使用一次的时候。
Lambda表达式语法结构
Lambda表达式通常使用(param)->(body)语法书写,基本格式如下:
常见的Lambda表达式如下:
对照上面的示例,我们再总结一下Lambda表达式的结构:
- Lambda表达式可以有0~n个参数。
- 参数类型可以显式声明,也可以让编译器从上下文自动推断类型。如(int x)和(x)是等价的。
- 多个参数用小括号括起来,逗号分隔。一个参数可以不用括号。
- 没有参数用空括号表示。
- Lambda表达式的正文可以包含零条,一条或多条语句,如果有返回值则必须包含返回值语句。如果只有一条可省略大括号。如果有一条以上则必须包含在大括号(代码块)中。
函数式接口
函数式接口(Functional Interface)是Java8对一类特殊类型的接口的称呼。这类接口只定义了唯一的抽象方法的接口(除了隐含的Object对象的公共方法),因此最开始也就做SAM类型的接口(Single Abstract Method)。
比如上面示例中的java.lang.Runnable就是一种函数式接口,在其内部只定义了一个void run()的抽象方法,同时在该接口上注解了@FunctionalInterface。
@FunctionalInterface注解是用来表示该接口要符合函数式接口的规范,除了隐含的Object对象的公共方法以外只可有一个抽象方法。当然,如果某个接口只定义一个抽象方法,不使用该注解也是可以使用Lambda表达式的,但是没有该注解的约束,后期可能会新增其他的抽象方法,导致已经使用Lambda表达式的地方出错。使用@FunctionalInterface从编译层面解决了可能的错误。
比如当注解@FunctionalInterface之后,写两个抽象方法在接口内,会出现以下提示:
通过函数式接口我们也可以得出一个简单的结论:可使用Lambda表达式的接口,只能有一个抽象方法(除了隐含的Object对象的公共方法)。
注意此处的方法限制为抽象方法,如果接口内有其他静态方法则不会受限制。
方法引用,双冒号操作
[方法引用]的格式是,类名::方法名。
像如ClassName::methodName或者objectName::methodName的表达式,我们把它叫做方法引用(Method Reference),通常用在Lambda表达中。
看一下示例:
再比如我们用函数式接口java.util.function.Function来实现一个String转Integer的功能,可以如下写法:
根据Function接口的定义Function<T,R>,其中T表示传入类型,R表示返回类型。具体就是实现了Function的apply方法,在其方法内调用了Integer.parseInt方法。
通过上面的讲解,基本的语法已经完成,以下内容通过实例来逐一演示在不同的场景下如何使用。
Runnable线程初始化示例
Runnable线程初始化是比较典型的应用场景。
通常都会把lambda表达式内部变量的名字起得短一些,这样能使代码更简短。
事件处理示例
Swing API编程中经常会用到的事件监听。
列表遍历输出示例
传统遍历一个List,基本上都使用for循环来遍历,Java8之后List拥有了forEach方法,可配合lambda表达式写出更加简洁的方法。
函数式接口示例
在上面的例子中已经看到函数式接口java.util.function.Function的使用,在java.util.function包下中还有其他的类,用来支持Java的函数式编程。比如通过Predicate函数式接口以及lambda表达式,可以向API方法添加逻辑,用更少的代码支持更多的动态行为。
其中filter方法中的写法还可以进一步简化:
如果不需要“符合条件的内容:”字符串的拼接,还能够进一步简化:
如果将调用filter方法的判断条件也写在一起,test方法中的内容可以通过一行代码来实现:
如果需要同时满足两个条件或满足其中一个即可,Predicate可以将这样的多个条件合并成一个。
Stream相关示例
在《JAVA8 STREAM新特性详解及实战》一文中已经讲解了Stream的使用。你是否发现Stream的使用都离不开Lambda表达式。是的,所有Stream的操作必须以Lambda表达式为参数。
以Stream的map方法为例:
更多的使用实例可参看Stream的《JAVA8 STREAM新特性详解及实战》一文。
Lambda表达式与匿名类的区别
- 关键词的区别:对于匿名类,关键词this指向匿名类,而对于Lambda表达式,关键词this指向包围Lambda表达式的类的外部类,也就是说跟表达式外面使用this表达的意思是一样。
- 编译方式:Java编译器编译Lambda表达式时,会将其转换为类的私有方法,再进行动态绑定,通过invokedynamic指令进行调用。而匿名内部类仍然是一个类,编译时编译器会自动为该类取名并生成class文件。
其中第一条,以Spring Boot中ServletWebServerApplicationContext类的一段源码作为示例:
其中,这里的this指向的就是getSelfInitializer方法所在的类。
小结
至此,Java8 Lambda表达式的基本使用已经讲解完毕,最关键的还是要勤加练习,达到熟能生巧的使用。当然,刚开始可能需要一个适应期,在此期间可以把本篇文章收藏当做一个手册拿来参考。
原文链接:《Java8 Lambda表达式详解手册及实例》
本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。
java中,某方法中含有接口做参数,为什么创建一个接口的实现类可以代替该接口作为该方法的参数
将类当成参数传入方法,其实就是将类的对象传入方法,如果是抽象类,其实就是将抽象类的子类的对象传入方法,如果是接口,其实就是将接口实现类的对象传入方法。
因为抽象类和接口是不能实例化成对象的,所以必须找它们的子类或实现类
JAVA:将类、抽象类、接口当成方法的参数传入
将类当成参数传入方法,其实就是将类的对象传入方法,如果是抽象类,其实就是将抽象类的子类的对象传入方法,如果是接口,其实就是将接口实现类的对象传入方法。
因为抽象类和接口是不能实例化成对象的,所以必须找它们的子类或实现类
1. 普通类对象当成方法参数传入
public class Person{ public void eat(){ System.out.println("吃饭"); } } //方法 public static void operatePerson(Person p){ p.eat(); } //main调用方法,将person对象传入方法中 operatePerson(new person());
2. 抽象类作为方法参数传入
//抽象类 public abstract class Animal{ public abstratc void eat(); } //Animal的子类 public class Cat extends Animal{ //重写方法 public void eat(){ System.out.println("猫吃鱼"); } } //main中的方法,这里要求传入Animal类,但Animal的类是一个抽象类,不能实例对象,所以只能传其子类。 function static void operateAnimal(Animal a){ a.eat(); } //方法的调用 Animal a = new Cat(); operateAnimal(new cat());或operateAnimal(a); 当然,还要以通过内部类一次性传参 operateAnimal( new Animal(){ //重写animal抽象方法 public void eat(){System.out.println("猫还吃饭");} } ); 要求传入父类对象,但可以传入任意子类对象,这样就使得扩展性得到提高 operateAnimal(new Cat()); operateAnimal(new Dog()); operateAnimal(new Bird()); ...... 传入什么类,就调用什么类的功能,这就是多态的实现。
3. 接口实现类的对象当成方法参数传入
public interface Smoking{ public abstract void smoking(); } //实现类 public class Student implements Smoking{ //实现接口方法 public void smoking(){ System.out.println("抽烟中...."); } } //main中定义方法,将接口实现类当方法参数传入 public static void operateSmoking(Smoking s){ s.smoking(); } //方法的调用 Smoking s = new Student(); operateSmoking(s);或operateSmoking(new Student()); 或用内部类传入 operateSmoking( new Smoking(){ //重写接口方法 public void smoking(){ System.out.println("不准吸烟"); } } );
总结:类当成方法参数传入,其实就是传对象。抽象类和接口其实就是传其子类或实现类的对象。