Lambda

前言

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());  //编译会出错 
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Lambda是java8出的新特性,之前很少用Lambda表达式写代码,现在慢慢习惯用Lambda表达式,并且记得...
    安仔夏天勤奋阅读 3,345评论 0 1
  • 注:之前关于Java8的认知一直停留在知道有哪些修改和新的API上,对Lambda的认识也是仅仅限于对匿名内部类的...
    mualex阅读 7,850评论 1 4
  • Java8-Lambda Java 8 的最大变化是引入了 Lambda 表达式——一种紧凑的、传递行为的方式 引...
    Cool_Pomelo阅读 2,948评论 0 5
  • 原文链接: Lambdas 原文作者: shekhargulati 译者: leege100 lambda表达式是...
    忽来阅读 11,650评论 8 129
  • 自人类社会诞生以来,财富的形式也在不断的改变和进化。最初的人类生活只能满足于生存,因而财富的形式是各种能让人们活下...
    康斯坦丁tony阅读 4,310评论 0 0

友情链接更多精彩内容