Java 8系列Lambda 表达式,一看就废

Posted 善良勤劳勇敢而又聪明的老杨

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java 8系列Lambda 表达式,一看就废相关的知识,希望对你有一定的参考价值。

热门系列:


目录

1.前言

1.1 为什么 Java 需要 Lambda 表达式?

2.正文

2.1 Lambda的编写方式

2.2Lambda 表达式的结构

2.3函数式接口

2.4方法引用

3.总结


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 Lambda表达式一看就会

Java 8系列Java日期时间的新主宰者:LocalDateLocalTimeLocalDateTimeZonedDateTime

Java 8系列全网最通俗易懂的Java8版本新特性知识汇总,看完不懂你捶我

Java 8系列全网最通俗易懂的Java8版本新特性知识汇总,看完不懂你捶我