深入理解Java8Lambada表达式

1. lambada表达式简介

  Lambda 表达式是一种匿名函数(对 Java 而言这并不完全正确,但现在姑且这么认为),简单地说,它是没有声明的方法,也即没有访问修饰符、返回值声明和名字。
  你可以将其想做一种速记,在你需要使用某个方法的地方写上它。当某个方法只使用一次,而且定义很简短,使用这种速记替代之尤其有效,这样,你就不必在类中费力写声明与方法了。

Java 中的 Lambda 表达式通常使用 (argument) -> (body) 语法书写,例如:

(arg1, arg2...) -> { body }
(type1 arg1, type2 arg2...) -> { body }

常见的写法如下

(int a, int b) -> {  return a + b; }
() -> System.out.println("Hello World");
(String s) -> { System.out.println(s); }
() -> 42
() -> { return 3.1415 };

2. Lambda 表达式的结构

  • 一个 Lambda 表达式可以有零个或多个参数
  • 参数的类型既可以明确声明,也可以根据上下文来推断。例如:(int a)与(a)效果相同
  • 所有参数需包含在圆括号内,参数之间用逗号相隔。例如:(a, b) 或 (int a, int b) 或 (String a, int b, float c)
  • 空圆括号代表参数集为空。例如:() -> 42
  • 当只有一个参数,且其类型可推导时,圆括号()可省略。例如:a -> return a*a
  • Lambda 表达式的主体可包含零条或多条语句
  • 如果 Lambda 表达式的主体只有一条语句,花括号{}可省略。匿名函数的返回类型与该主体表达式一致
  • 如果 Lambda 表达式的主体包含一条以上语句,则表达式必须包含在花括号{}中(形成代码块)。匿名函数的返回类型与代码块的返回类型一致,若没有返回则为空

3. 函数式接口

  函数式接口是只包含一个抽象方法声明的接口.java.lang.Runnable 就是一种函数式接口,在 Runnable 接口中只声明了一个方法 void run(),相似地,ActionListener 接口也是一种函数式接口,我们使用匿名内部类来实例化函数式接口的对象,有了 Lambda 表达式,这一方式可以得到简化。
每个 Lambda 表达式都能隐式地赋值给函数式接口,例如,我们可以通过 Lambda 表达式创建 Runnable 接口的引用。

Runnable r = () -> System.out.println("hello world");

编译器会把lambada表达式当做一个函数来编译.

4. 常见的函数式接口

4.1 Runnable

Runnable runnable = ()->{
    System.out.println(Thread.currentThread().getName());
};
        
Thread thread = new Thread(runnable);
thread.start();
  • Runnable只有一个run函数,且run函数没有参数

4.2 Consumer接口

Consumer接口接收一个参数,不返回参数

public static void consumerFun(int value, Consumer<Integer> c) {
    c.accept(value);
}
//调用
consumerFun(1,(value)->{
    System.out.println(value);
});

4.3 BinConsumer接口

与Consumer接口一样,只不过接收两个参数,返回0个参数

public static void binConsumerFun(String a, String b, BiConsumer<String, String> binc) {
    binc.accept(a, b);
}
//调用
binConsumerFun("hello", "maskwang", (a,b)->{
    System.out.println(a+b)
});

4.4 Predication

作用接收一个参数,返回一个boolean值

public static boolean predicateFun(int value, Predicate<Integer> pre) {
    return pre.test(value);
}
//调用
System.out.println(predicateFun(3, x->x==3));

4.5 Supplier

作用是接收0个参数,返回一个值

public static int supplierFun(Supplier<Integer> supplier) {
    return supplier.get();
}
//调用
System.out.println(supplierFun(()->1));

4.5 Comparator

lambada表达式实现Comparator

List<Integer> list = new ArrayList<>();
list.add(1);
list.add(3);
list.add(2);
list.sort((a,b) -> {return a>b?-1:1;});
 //另一种方式
list.forEach(System.out::println)

4.6 集合的操作

实现集合的遍历

List<Integer> list = new ArrayList<>();
list.add(1);
list.add(3);
list.add(2);
list.forEach((value)->System.out.println(value));

5. Lambda表达式与匿名内部类的联系和区别

联系:

  • lambda表达式创建的对象与匿名内部类生成的对象一样,可以直接调用接口中继承的默认方法。

区别:

  • 匿名内部类可以为任意接口创建实例,不管接口中包含多少个抽象方法,只要在匿名内部类中实现所有抽象方法即可。 但在lambda表达式中只能为函数式接口创建实例。

  • 匿名内部类可以为抽象类甚至普通类创建实例;但lambda表达式只能为函数式接口创建实例。

  • 匿名内部类实现的抽象方法可以允许调用接口中定义默认方法。但lambda表达式的代码块不允许调用接口中定义默认方法。

6. 总结

Lambda 表达式赋予了 Java 相较于其他函数式编程语言缺失的特性,结合虚拟扩展方法之类的特性,Lambda 表达式能写出一些极好的代码。希望能在以后写代码的过程中,把这些用上去,使整个代码看起来很简洁.

ps: ::运算符是方法的引用,用这种方式可以简单代替Lambada

String::valueOf         x -> String.valueOf(x)
Object::toString        x -> x.toString()
x::toString         () -> x.toString()
ArrayList::new      () -> new ArrayList<>()
发布了5 篇原创文章 · 获赞 0 · 访问量 269

猜你喜欢

转载自blog.csdn.net/robingdo/article/details/104964006