Lambda

作者: __元昊__ | 来源:发表于2020-04-29 17:13 被阅读0次

    前言

    Java11前两天都发布了,而自己Java8还没搞明白,真是羞为称自己为Java程序员。今天就让我们来通俗易懂的聊一聊Java8中的Lambda表达式~

    简单上手

    u=20823480,933902871&fm=173&app=25&f=JPEG.jpg

    通过Lamda表达式,可以变换为:

    new Thread(() -> System.out.println("thread"));
    

    针对这种实行,我们怎么理解呢?其实很简单,上看一下上述lambda表达式的语法:() -> {} (): 括号就是接口方法的括号,接口方法如果有参数,也需要写参数。只有一个参数时,括号可以省略。-> : 分割左右部分的,没啥好讲的。{} : 要实现的方法体。只有一行代码时,可以不加括号,可以不写return。

    不过看到这里我相信有些小伙伴已经许意识到了,如果接口中有多个方法时,那么按照上面的逻辑lambda表达式恐怕没办法表示了。的确是这样,并非任何接口都支持lambda表达式。

    而适用于lambda表达式的接口称之为函数型接口。说白了,函数型接口就是只有一个抽象方法的接口。

    函数式接口

    其实之前在讲Lambda表达式的时候提到过,所谓的函数式接口,当然首先是一个接口,然后就是在这个接口里面只能有一个抽象方法。

    这种类型的接口也称为SAM接口,即Single Abstract Method interfaces。

    1.1、函数式接口基本语法

    它们主要用在Lambda表达式和方法引用(实际上也可认为是Lambda表达式)上。

    如定义了一个函数式接口如下:


    u=2729990849,1446984129&fm=173&app=25&f=JPEG.jpg

    那么就可以使用Lambda表达式来表示该接口的一个实现(注:JAVA 8 之前一般是用匿名类实现的):

    GreetingService greetService1 = message -> System.out.println("Hello " + message);
    

    1.2、FunctionalInterface注解

    关于@FunctionalInterface注解Java 8为函数式接口引入了一个新注解@FunctionalInterface,主要用于编译级错误检查,加上该注解,当你写的接口不符合函数式接口定义的时候,编译器会报错。

    正确例子,没有报错:


    u=2729990849,1446984129&fm=173&app=25&f=JPEG.jpg

    1.3、用法提醒

    ERROR:接口中包含了两个抽象方法,违反了函数式接口的定义,IDE会直接报错。

    Tips:加不加@FunctionalInterface对于接口是不是函数式接口没有影响,该注解知识提醒编译器去检查该接口是否仅包含一个抽象方法

    1.4、默认方法

    函数式接口里是可以包含默认方法,因为默认方法不是抽象方法,其有一个默认实现,所以是符合函数式接口的定义的;


    u=2547916558,3030871120&fm=173&app=25&f=JPEG.jpg

    1.5、静态方法

    函数式接口里是可以包含静态方法,因为静态方法不能是抽象方法,是一个已经实现了的方法,所以是符合函数式接口的定义的;

    如下代码不会报错:


    u=2268955634,2131712240&fm=173&app=25&f=JPEG.jpg

    1.6、Object里的public方法

    函数式接口里是可以包含Object里的public方法,这些方法对于函数式接口来说,不被当成是抽象方法(虽然它们是抽象方法);因为任何一个函数式接口的实现,默认都继承了Object类,包含了来自java.lang.Object里对这些抽象方法的实现;

    如下代码不会报错:


    u=1494202482,2752892781&fm=173&app=25&f=JPEG.jpg

    进阶

    有了上面的基础,我们稍稍聊一些深入的lambda表达式。lambda表达式还有两种简化代码的手段,它们是方法引用、构造引用。

    方法引用是什么呢?如果我们要实现接口的方法与另一个方法A类似,(这里的类似是指参数类型与返回值部分相同),我们直接声明A方法即可。也就是,不再使用lambda表达式的标准形式,改用高级形式。无论是标准形式还是高级形式,都是lambda表达式的一种表现形式。

    Function function1 = (x) -> x;Function function2 = String::valueOf;
    

    对比Function接口的抽象方法与String的value方法,可以看到它们是类似的。


    u=1323023801,4053429754&fm=173&app=25&f=JPEG.jpg

    方法引用的语法:

    对象::实例方法类::静态方法类::实例方法

    前两个很容易理解,相当于对象调用实例方法,类调用静态方法一样。只是第三个需要特殊说明。

    Compare<Boolean> c = String::equals;
    

    也就是“类::实例方法”的形式。

    构造引用

    u=1618868181,3291930072&fm=173&app=25&f=JPEG.jpg

    提炼一下构造引用的语法:类名::new


    u=492617495,3453585447&fm=173&app=25&f=JPEG.jpg

    变量作用域

    lambda 表达式只能引用标记了 final 的外层局部变量,这就是说不能在 lambda 内部修改定义在域外的局部变量,否则会编译错误。

    在 Java8Tester.java 文件输入以下代码:

    public class Java8Tester {
     
       final static String salutation = "Hello! ";
       
       public static void main(String args[]){
          GreetingService greetService1 = message -> 
          System.out.println(salutation + message);
          greetService1.sayMessage("Runoob");
       }
        
       interface GreetingService {
          void sayMessage(String message);
       }
    }
    

    执行以上脚本,输出结果为:

    Hello! Runoob
    

    我们也可以直接在 lambda 表达式中访问外层的局部变量:

    public class Java8Tester {
        public static void main(String args[]) {
            final int num = 1;
            Converter<Integer, String> s = (param) -> System.out.println(String.valueOf(param + num));
            s.convert(2);  // 输出结果为 3
        }
     
        public interface Converter<T1, T2> {
            void convert(int i);
        }
    }
    

    lambda 表达式的局部变量可以不用声明为 final,但是必须不可被后面的代码修改(即隐性的具有 final 的语义)

    int num = 1;  
    Converter<Integer, String> s = (param) -> System.out.println(String.valueOf(param + num));
    s.convert(2);
    num = 5;  
    //报错信息:Local variable num defined in an enclosing scope must be final or effectively 
     final
    

    在 Lambda 表达式当中不允许声明一个与局部变量同名的参数或者局部变量。

    String first = "";  
    Comparator<String> comparator = (first, second) -> Integer.compare(first.length(), second.length());  //编译会出错 
    

    相关文章

      网友评论

          本文标题:Lambda

          本文链接:https://www.haomeiwen.com/subject/yrjnwhtx.html