Java 8系列Lambda 表达式,一看就废
Posted 善良勤劳勇敢而又聪明的老杨
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java 8系列Lambda 表达式,一看就废相关的知识,希望对你有一定的参考价值。
热门系列:
-
【Java 8系列】收集器Collector与工具类Collectors
-
【Java 8系列】Stream详解,看这一篇就够啦
-
【Java 8系列】Java日期时间的新主宰者:LocalDate、LocalTime、LocalDateTime、ZonedDateTime
-
【Java 8系列】Java开发者的判空利器 – Optional
-
程序人生,精彩抢先看
目录
1.前言
Lambda 表达式,也可称为闭包,它是推动 Java 8 发布的最重要新特性。Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中)。使用Lambda 表达式可以使代码变的更加简洁紧凑。
Lambda 表达式是一种匿名函数(对 Java 而言这并不完全正确,但现在姑且这么认为),简单地说,它是没有声明的方法,也即没有访问修饰符、返回值声明和名字。
1.1 为什么 Java 需要 Lambda 表达式?
Lambda 表达式为 Java 添加了缺失的函数式编程特点,使我们能将函数当做一等公民看待。尽管不完全正确,我们很快就会见识到 Lambda 与闭包的不同之处,但是又无限地接近闭包。在支持一类函数的语言中,Lambda 表达式的类型将是函数。但是,在 Java 中,Lambda 表达式是对象,他们必须依附于一类特别的对象类型——函数式接口(functional interface)。
2.正文
2.1 Lambda的编写方式
Java 中的 Lambda 表达式通常使用 (argument) -> (body)
语法书写,例如:
(arg1, arg2...) -> body
(type1 arg1, type2 arg2...) -> body
2.2Lambda 表达式的结构
Lambda 表达式的结构一般有如下几种:
- 一个 Lambda 表达式可以有零个或多个参数。例如:
//无参数无返回值
() -> System.out.println("test");
//无参数有返回值
() ->
int a = 1;
int b = 2;
return a * b;
;
//有参数无返回值
(String s) -> System.out.println(s);
//有参数有返回值
(int a, int b) -> return a + b;
- 参数的类型既可以明确声明,也可以根据上下文来推断。例如:
@FunctionalInterface
public interface LambdaMethodService
int add(int x, int y);
//在main中调用
public static void main(String[] args)
LambdaMethodService service = (a,b)-> a+b;
//上下两种写法效果一样
//LambdaMethodService service = (int a,int b)->a+b;
service.add(1,2);
- 所有参数需包含在圆括号内,参数之间用逗号相隔。例如:
(a, b)
或(int a, int b)
或(String a, int b, float c)
- 空圆括号代表参数集为空。例如:
() -> 42
- 当只有一个参数,且其类型可推导时,圆括号()可省略。例如:
a -> return a*a
- Lambda 表达式的主体可包含零条或多条语句
- 如果 Lambda 表达式的主体只有一条语句,花括号可省略。匿名函数的返回类型与该主体表达式一致
- 如果 Lambda 表达式的主体包含一条以上语句,则表达式必须包含在花括号中(形成代码块)。匿名函数的返回类型与代码块的返回类型一致,若没有返回则为空
2.3函数式接口
刚才上面有提到,java的Lambda 表达式需要依赖于函数式接口;那么何为函数式接口?
在 Java 中,Marker(标记)类型的接口是一种没有方法或属性声明的接口,简单地说,marker 接口是空接口。相似地,函数式接口是只包含一个抽象方法声明的接口。(重点:敲黑板,记好,这个要考!!!)
java.lang.Runnable
就是一种函数式接口,在 Runnable 接口中只声明了一个方法 void run()
,我们使用匿名内部类来实例化函数式接口的对象,有了 Lambda 表达式,这一方式可以得到简化。例如:
Runnable r = () -> System.out.println("hello world");
new Thread(
() -> System.out.println("hello world")
).start();
@FunctionalInterface 是 Java 8 新加入的一种接口,用于指明该接口类型声明是根据 Java 语言规范定义的函数式接口。Java 8 还声明了一些 Lambda 表达式可以使用的函数式接口,当你注释的接口不是有效的函数式接口时,可以使用 @FunctionalInterface 解决编译层面的错误。例如:
@FunctionalInterface
public interface WorkerInterface
public void doSomeWork();
但是,根据定义,函数式接口只能有一个抽象方法,如果你尝试添加第二个抽象方法,将抛出编译时错误。例如你在上面再添加一个方法的话:
@FunctionalInterface
public interface WorkerInterface
public void doSomeWork();
public void doSomeMoreWork();
则会报错:
Unexpected @FunctionalInterface annotation
@FunctionalInterface ^ WorkerInterface is not a functional interface multiple
non-overriding abstract methods found in interface WorkerInterface 1 error
2.4方法引用
双冒号(::
)操作符是 Java 中的方法引用。 当们使用一个方法的引用时,目标引用放在 ::
之前,目标引用提供的方法名称放在 ::
之后,即 目标引用::方法
。比如:
Person::getAge;
再结合lambda,例如,要创建一个比较器,以下语法就足够了:
Comparator c = (Person p1, Person p2) -> p1.getAge().compareTo(p2.getAge());
Comparator c = Comparator.comparing(Person::getAge);
3.总结
其实呢,我个人感觉这种Lambda表达式的写法,虽然简化了代码量,提高了开发效率。但是,在代码的可读性上,不是很友好!当然,也可能是对于java中一种新的表现形式的陌生感所导致!所以,不知道各位觉得Lambda好不好用呢?欢迎下方留言讨论!
写到这里,后面我将还会分享和Lambda紧紧关联的java8的另一个新特性:Stream,感兴趣的朋友,请关注我吧!!
以上是关于Java 8系列Lambda 表达式,一看就废的主要内容,如果未能解决你的问题,请参考以下文章
Java 8系列收集器Collector与工具类Collectors
Java 8系列Java开发者的判空利器 -- Optional
Java 8系列Java日期时间的新主宰者:LocalDateLocalTimeLocalDateTimeZonedDateTime