Lambda表达式是JDK 8开始后的一种新语法形式。

作用:简化匿名内部类的代码写法

简化格式

(匿名内部类被重写方法的形参列表) -> {
重写方法
}

Lambda表达式只能简化函数式接口的匿名内部类的写法形式

什么是函数式接口?

  • 首先必须是接口、其次接口中有且仅有一个抽象方法的形式
  • 通常会在接口上加上一个@FunctionalInterface注解,标记该接口必须是满足函数式接口

如何使用Lambda?

我们将根据下面三个问题来帮助大家理解和使用Lambda

背景:我们自定义了一个man的类,创建了一个man的List。

class man {
public int age;
public char sex;
public double socre; public man(int age, char sex, double score) {
this.age = age;
this.sex = sex;
this.score = score;
}
}

问题一

现需要对这个list根据人的年龄进行排序

要实现排序的功能,可以直接调用List对象自带的sort方法完成,但是需要man先实现Comparator的接口并重写compare方法,编译器才能比较两个不同man的大小。但是要更改原始类的代码,会比较麻烦,如果以后要对人的分数进行排序,那就又要更改的类的源码,这样操作很不方便。

sort(Comparator<? super E> c)方法可以直接传入一个Comparator对象,我们可以直接改写compare方法就可以实现比较。

第一种写法

public class lambdaTry {
public static void main(String[] args) {
List<man> humans = new ArrayList<>();
humans.add(new man(19, 'g', 98.0));
humans.add(new man(18, 'b', 95.0));
humans.add(new man(20, 'b', 96.0));
humans.add(new man(17, 'g', 97.0)); humans.sort(new Comparator<man>() {
@Override
public int compare(man o1, man o2) {
return o1.age - o2.age;
}
});
}
}

第二种写法

Lambda

我们知道Lambda是用来简化函数式接口的匿名内部类,且Comparator满足函数式接口的两个条件:

  • 首先必须是接口、其次接口中有且仅有一个抽象方法的形式
  • @FunctionalInterface注解
@FunctionalInterface
public interface Comparator<T> {
int compare(T o1, T o2);
...
}

因此我们可以对上述的源码进行改写成Lambda格式

public class lambdaTry {
public static void main(String[] args) {
List<man> humans = new ArrayList<>();
humans.add(new man(19, 'g', 98.0));
humans.add(new man(18, 'b', 95.0));
humans.add(new man(20, 'b', 96.0));
humans.add(new man(17, 'g', 97.0)); humans.sort((man o1, man o2) -> {
return o1.age - o2.age;
});
}
}

改写过后代码简洁了很多。但是还可以继续简写。

Lambda表达式的省略写法

  1. 参数类型可以不写
  2. 如果只有一个参数,参数类型可以省略,同时()也可以省略
  3. 如果Lambda表达式的方法块中代码只有一行,可以省略大括号,同时省略分号。
  4. 在条件三的基础上,如果这行代码是return语句,必须省略return。

第三种写法

Lambda简写

可以看到,此表达式满足省略写法的条件,可以继续简写成如下格式。只需要一行语句就能完成

public class lambdaTry {
public static void main(String[] args) {
List<man> humans = new ArrayList<>();
humans.add(new man(19, 'g', 98.0));
humans.add(new man(18, 'b', 95.0));
humans.add(new man(20, 'b', 96.0));
humans.add(new man(17, 'g', 97.0)); humans.sort((o1, o2) -> o1.age - o2.age);
}
}

问题二

将List转换为数组

我们知道List接口有一个方法toArray方法可以实现将其转换为数组。

JDK11之后,提供了这样的一个方法,提供了一个函数式接口来让我们转换

default <T> T[] toArray(IntFunction<T[]> generator) {
return toArray(generator.apply(0));
}

IntFunction函数式接口是从JDK8之后实现的,内部只有一个apply抽象方法,是一个标准的函数式接口

@FunctionalInterface
public interface IntFunction<R> {
R apply(int value);
}

我们可以直接用lambda,完成数组的转换

public class lambdaTry {
public static void main(String[] args) {
List<man> humans = new ArrayList<>();
humans.add(new man(19, 'g', 98.0));
humans.add(new man(18, 'b', 95.0));
humans.add(new man(20, 'b', 96.0));
humans.add(new man(17, 'g', 97.0)); // 原本写法
// man[] mans = humans.toArray(new IntFunction<man[]>() {
// @Override
// public man[] apply(int value) {
// return new man[value];
// }
// }); // lambda写法
man[] mans = humans.toArray(value -> new man[value]); // 实际上用不上这样的写法,只是为了举例说明
// man[] mans = humans.toArray(new man[0]);
// man[] mans = humans.toArray(man[]::new);
// 上面两种写法都可以,传值进去的size为0不影响实际的转换,具体可以看ArrayList的toArray重写方法 }
}

问题三

输出年龄大于18的男同学的成绩

可以用forEach方法快捷实现,forEach方法来自于Iterable接口

default void forEach(Consumer<? super T> action) {
Objects.requireNonNull(action);
for (T t : this) {
action.accept(t);
}
}

再看Consumer接口,也是一个函数式接口

@FunctionalInterface
public interface Consumer<T> {
void accept(T t);
...
}

具体实现

public class lambdaTry {
public static void main(String[] args) {
List<man> humans = new ArrayList<>();
humans.add(new man(19, 'g', 98.0));
humans.add(new man(18, 'b', 95.0));
humans.add(new man(20, 'b', 96.0));
humans.add(new man(17, 'g', 97.0)); // humans.forEach(new Consumer<>() {
// @Override
// public void accept(man man) {
// if (man.age >= 18 && man.sex == 'g') {
// System.out.println(man.score);
// }
// }
// }); humans.forEach(man -> {
if (man.age >= 18 && man.sex == 'g') {
System.out.println(man.score);
}
});
}
}

有时Lambda还可以继续简写成方法引用(method reference)

方法引用

方法引用通过方法的名字来指向一个方法。

方法引用可以使语言的构造更紧凑简洁,减少冗余代码。

方法引用使用一对冒号 ::

主要分为四种:

  • 构造器引用 Class::new

    man[] mans = humans.toArray(man[]::new);
  • 静态方法引用 Class::static_method

    打印每个man(需要在man内重写toString)

    humans.forEach(System.out::println)
  • 特定类的任意对象的方法引用 Class::method

  • 特定对象的方法引用 instance::method

Java Lambda详解的更多相关文章

  1. Java集合详解3:一文读懂Iterator,fail-fast机制与比较器

    <Java集合详解系列>是我在完成夯实Java基础篇的系列博客后准备开始写的新系列. 这些文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查 ...

  2. Java 集合详解 | 一篇文章解决Java 三大集合

    更好阅读体验:Java 集合详解 | 一篇文章搞定Java 三大集合 好看的皮囊像是一个个容器,有趣的灵魂像是容器里的数据.接下来讲解Java集合数据容器. 文章篇幅有点长,还请耐心阅读.如只是为了解 ...

  3. Java内部类详解

    Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就 ...

  4. 黑马----JAVA迭代器详解

    JAVA迭代器详解 1.Interable.Iterator和ListIterator 1)迭代器生成接口Interable,用于生成一个具体迭代器 public interface Iterable ...

  5. C++调用JAVA方法详解

    C++调用JAVA方法详解          博客分类: 本文主要参考http://tech.ccidnet.com/art/1081/20050413/237901_1.html 上的文章. C++ ...

  6. Java虚拟机详解----JVM常见问题总结

    [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...

  7. [转] Java内部类详解

    作者:海子 出处:http://www.cnblogs.com/dolphin0520/ 本博客中未标明转载的文章归作者海子和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置 ...

  8. Java面向对象详解

    Java面向对象详解 前言:接触项目开发也有很长一段时间了,最近开始萌发出想回过头来写写以前学 过的基础知识的想法.一是原来刚开始学习接触编程,一个人跌跌撞撞摸索着往前走,初学的时候很多东西理解的也懵 ...

  9. java 乱码详解_jsp中pageEncoding、charset=UTF -8"、request.setCharacterEncoding("UTF-8")

    http://blog.csdn.net/qinysong/article/details/1179480 java 乱码详解__jsp中pageEncoding.charset=UTF -8&quo ...

随机推荐

  1. 【windows 操作系统】协程

    ◆ 协程 由程序员手动切换. 属于线程,是在线程里面跑的,因此协程又称微线程和线程等.由于不用进行线程上下文切换,因此效率会更高. 资源: 拥有自己的栈空间,大小动态调整. 安全性: 拥有自己的寄存器 ...

  2. 【基础知识】CPU原理之减法、乘法和除法

    中介绍了布尔逻辑.数学和电路的关系,我们也得到了与门.或门.非门.或非门.与非门.异或门等门电路以及一个加法器,并且了解了计算机是如何做加法的,这篇文章介绍一下计算机是如何做减法以及乘除法的. 0x0 ...

  3. C语言刷“矩阵”类题目(2维矩阵/2级指针)

    566. 重塑矩阵 int** matrixReshape(int** mat, int matSize, int* matColSize, int r, int c, int* returnSize ...

  4. npm cnpm yarn 安装

    安装node.js,其中已经集成了npm,可以将npm切换到国内镜像 $ npm config set registry https://registry.npm.taobao.org -- 配置后可 ...

  5. 『现学现忘』Docker相关概念 — 1、云计算概念

    目录 1.云计算的概念 2.示例说明云计算 3.小故事说明云计算 "云计算"这个词,相信大家都非常熟悉. 作为信息科技发展的主流趋势,它频繁地出现在我们的眼前.伴随它一起出现的,还 ...

  6. Python语法3

    目录 模块 Python程序架构 第三方工具包: 创建packet包 模块导入方式 异常处理 六种典型异常 异常处理 自定义异常 模块 Python程序架构 Python源代码文件:*.py 一个py ...

  7. 扩展auth_user字段、BBS需求分析、创建BBS数据库、注册页面搭建与用户头像展示及Ajax提交数据

    昨日内容回顾 csrf跨站请求 1. SQL注入 2. xss攻击 3. csrf跨站请求 4. 密码加密(加盐) '''django中默认有一个中间件来验证csrf''' # 只针对post请求才验 ...

  8. CentOS 8: yum仓库配置

    在CentOS 8中,使用yum时出现错误,镜像列表中没有url,类似如下: Error: Failed to download metadata for repo 'appstream': Cann ...

  9. aria2 源码解析专题—— (一)基础架构

    此文章是这个专题的开篇,由于初入 C++ 的大门,所以想着拿个项目来看看,凑巧有点基础,又想学习一下在unix上的编程,所以就找了 aria2 这个库来看看源码,希望能学到一些东西. 言归正传,今天把 ...

  10. GoLang设计模式21 - 装饰模式

    装饰器模式是一种结构型设计模式.通过装饰器模式可以为一个对象添加额外的功能而不需对其作出调整. 还是通过具体的案例来了解装饰器模式:假设我们开了一家披萨店,现在店里主营两款披萨: 素食狂披萨(Vegg ...