Lambda表达式详解

Lambda表达式的使用

前置介绍

函数式接口:如果一个接口中,只声明了一个抽象方法,则此接口就称为函数式接口。
lambda表达式的本质:作为函数式接口的实例
何时使用:当需要对一个函数式接口实例化的时候,可以使用lambda表达式。

Lambda表达式的基本语法

  1. 举例:(o1,o2) -> Integer.compare(o1,o2);
  2. 格式:
    :lambda操作符 或 箭头操作符
    左边:lambda形参列表 (接口中的抽象方法的形参列表)
    右边:lambda体 (重写的抽象方法的方法体)

    六种语法格式

    语法格式一:无参,无返回值

    Runnable r1 = () -> {System.out.println("hello lambda!");};

    语法格式二:Lambda需要一个参数,但是没有返回值

    Consumer con = (String str) -> {System.out.println(str);};

    语法格式三:数据类型可以省略,因为可由编译器推断得出,称为“类型推断”

    Consumer con = (str) -> {System.out.println(str);};

    语法格式四:Lambda若只需要一个参数时,参数的小括号可以省略

    Consumer con = str -> {System.out.println(str);};

    语法格式五:Lambda若只需要两个或以上的参数,多条执行语句,并且可以有返回值

    Consumer con = (x,y) -> {
         System.out.println("实现函数式接口方法!");
         return Integer.compare(x,y);
    };

    语法格式六:当Lambda体只有一条语句时,return与大括号若有,都可以省略

    Consumer con = (x,y) -> Integer.compare(x,y);

    总结六种情况:
    左边:lambda形参列表的参数类型可以省略(类型推断);如果lambda形参列表只有一个参数,其一对()也可以省略
    右边:lambda体应该使用一对{}包裹;如果lambda体只有一条执行语句(可能是return语句,省略这对{}和return关键字)

方法引用

  1. 使用情境:当要传递给lambda体的操作,已经有实现的方法了,可以使用方法引用!
  2. 方法引用,本质上就是lambda表达式,而lambda表达式作为函数式接口的实例,所以方法引用,也是函数式接口的实例。
  3. 使用格式:类(或对象)::方法名
  4. 具体分为如下的三种情况
    情况1 对象::非静态方法
    情况2 类::静态方法
    情况3 类::非静态方法
  5. 方法引用使用的要求:要求接口中的抽象方法的形参列表和返回值类型与方法引用的方法的形参列表和返回值类型相同!(针对于情况1和情况2)
    当函数式接口方法的第一个参数是需要引用方法的调用者,并且第二个参数是需要引用方法的参数(或无参数)时:ClassName::methodName(针对于情况3)
  6. 使用建议
    如果给函数式接口提供实例,恰好满足方法引用的使用情境,就可以考虑使用方法引用给函数式接口提供实例。如果不熟悉方法引用,还是可以使用lambda表达式
  7. 样例

    情况一:
    @Test
    public void test1() {
         Consumer con1 = str -> System.out.println(str);
         con1.accept("北京");
    
         System.out.println("***************");
         PrintStream ps = System.out;
         Consumer con2 = ps::println;
         con2.accept("beijing");
    }
    情况二:
    @Test
    public void test2() {
         Comparator com1 = (t1,t2) -> Integer.compare(t1,t2);
         System.out.println(com1.compare(12,21));
    
         System.out.println("***************");
    
         Comparator com2 = Integer::compare;
             System.out.println("com2.compare(12,3)");
    }
    情况三:
    @Test
    public void test3() {
         Comparator com1 = (s1,s2) -> s1.compareTo(s2);
         System.out.println(com1.compare("abc","abd"));
    
         System.out.println("***************");
    
         Comparator com2 = String::compareTo;
         System.out.println("com2.compare("abd","abm")");
    }

    构造器引用
    格式:类名::new
    要求:和方法引用类似,函数式接口的抽象方法形参列表和构造器的形参列表一致。抽象方法的返回值类型即为构造器所属的类的类型
    数组引用
    格式:数组类型[] :: new

你可能感兴趣的