2014年3月18日发布了JavaSE 8

不追求技术的新,追求技术的稳定

本质:Lambda 表达式是一个匿名函数

作用:简化代码,增强代码的表达力

Lambda 语法格式

// 格式1:无参无返回值
() -> System.out.println("Hello World!"); // 格式2:有参无返回值
(x) -> System.out.println(x); // 格式3:有参有返回值
(x) -> x * x; // 格式4:多参有返回值
(x, y) -> x + y; // 格式5:函数体包含多条语句
(x, y) -> {
System.out.println("加法运算");
return x + y;
}

Lambda 表达式中的参数的数据类型可以省略,JVM 编译器能够根据上下文推算出,即“类型推断”

两个例子

/** 1. Comparator **/
TreeSet<Integer> ts1 = new TreeSet<>(new Comparator<Integer>(){
@Override
public int compare(Integer i1, Integer i2) {
return Integer.compare(i1, i2);
}
});
// lambda 表达式
TreeSet<Integer> ts2 = new TreeSet<>((i1, i2) -> {
return Integer.compare(i1, i2);
});
// 等同于(使用方法引用还可以再次简化)
TreeSet<Integer> ts3 = new TreeSet<>((i1, i2) -> Integer.compare(i1, i2)); /** 2. Runnable */
Thread t1 = new Thread(new Runnable(){
@Override
public void run() {
System.out.println("当前线程:" + Thread.currentThread().getName());
}
});
// lambda 表达式
Thread t2 = new Thread(() -> {
System.out.println("当前线程:" + Thread.currentThread().getName());
});

函数式接口

!!Lambda 表达式需要函数式接口的支持

函数式接口:接口只有一个抽象方法

可以使用注解 @FunctionalInterface 修饰接口,检查是否是函数式接口

// 定义函数式接口
@FunctionalInterface
public interface Calculator<T> {
public T calculate(T x, T y);
} // 使用函数式接口
Calculator<Integer> calculatorAdd = (x, y) -> x + y;
Integer result = calculatorAdd.calculate(3, 5);

Java 内置了四大核心函数式接口:

消费型接口 Consumer<T>:消费一个参数对象

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

供给型接口 Supplier<T>:返回一个对象

@FunctionalInterface
public interface Supplier<T> {
T get();
}

函数型接口 Function<T, R>:传递一个参数,返回一个值

@FunctionalInterface
public interface Function<T, R> {
R apply(T t);
...
}

断定型接口 Predicate<T>:判断参数是否满足约束

@FunctionalInterface
public interface Predicate<T> {
boolean test(T t);
...
}

对于Java内置函数式接口建议结合 stream 方法理解,在这里了解即可

除了这4大核心函数式接口外,还有由它们延伸出的一些变种,比如二元消费型接口 BiConsumer<T, U>

public interface BiConsumer<T, U> {
void accept(T t, U u);
...
}

方法引用

将 lambda 体代码封装为方法,然后方法引用,再次简化代码。

方法引用格式:类名::方法名对象::方法名

温馨提示:

实际上,在开发工具 IDEA 中,会自动提示使用方法引用简化代码,你只需按 ALT+Eenter 快捷键,根据提示选择操作即可

如果你想要深入了解方法引用的使用原则,可以继续往下看。(即使不看也没大问题,有开发工具帮你优化)

使用方法引用改写 Comparator 例子中的 lambda 表达式

//
TreeSet<Integer> ts3 = new TreeSet<>((i1, i2) -> Integer.compare(i1, i2));
// 使用方法引用
TreeSet<Integer> ts4 = new TreeSet<>(Integer::compare);

(第一种情况)实现函数式接口方法的参数列表,必须和方法引用方法的参数列表保持一致

Comparator.compare(o1, o2) 的 o1, o2 与 Integer.compare(i1, i2) 中的 i1, i2 对应,所以才能够使用方法应用。

当函数式接口方法只有一个参数时(小例子):


@Test
public void test3() {
List<String> stringList = Arrays.asList("北京", "天津", "上海");
// `Consumer.accept(t)` 的参数 t 与 `System.out.println(o)` 的 o 对应
show(System.out::println, stringList);
}
// 自定义一个函数
void show(Consumer<String> consumer, List<String> list) {
for (String s : list) {
consumer.accept(s);
}
}

还有第二种情况

TreeSet<Integer> ts3 = new TreeSet<>((i1, i2) -> i1.compareTo(i2));
// 使用方法引用
TreeSet<Integer> ts4 = new TreeSet<>(Integer::compareTo);

Comparator.compare(o1, o2) 的 o1, o2 与 i1.compareTo(i2) 中 i1, i2 对应,这样也能使用方法引用。

(第二种情况)假设函数式接口方法参数有 (x1, x2),而方法实现是 x1.fun(x2) 这种形式,照样使用方法引用

如果理解了它们的规律,推而广之,可以试试抽象方法含有三个参数的情况。

准备工作:找一个三参数的函数式接口

	@Test
public void test4() {
String s = "Hello World";
Integer start = 0;
Integer length = 5;
String r1 = consume((o1, o2, o3) -> {
return o1.substring(o2, o3);
}, s, start, length);
System.out.println(r1); String r2 = consume(String::substring, s, start, length);
System.out.println(r2);
}
String consume(TripleFunction<String, Integer, Integer, String> tripleFunction,
String s,
Integer start,
Integer length) {
return tripleFunction.apply(s, start, length);
}
// 自定义三参数函数式接口
@FunctionalInterface
interface TripleFunction<T, U, E, R> {
R apply(T t, U u, E e);
}

这里 函数式接口 TripleFunction 的抽象方法 apply(T t, U u, E e)中的参数 t, u, e 与 s.substring(start, length) 中的 s,start, length 对应

小结:

设函数式接口抽象方法 abstractFun(n1, n2, n3, ..., nn)

**有方法fun(n1, n2, n3, ..., nn)n1.fun(n2, n3, ..., nn) ** 实现了 lambda 体的代码功能

就可使用方法引用 ClassName::fun

构造器引用

用法:


Function<Integer, MyClass> fun1 = (i) -> new MyClass(i);
// 使用构造器引用
Function<Integer, MyClass> fun2 = MyClass::new; Function<Integer, Integer[]> fun3 = (n) -> new Integer[n];
// 使用构造器引用
Function<Integer, Integer[]> fun4 = Integer[]::new;

函数式接口方法参数列表,必须和构造函数参数列表一致 (和方法应用的第一种情况相同)

Java8 新特性(一)- Lambda 表达式的更多相关文章

  1. java8新特性: lambda表达式:直接获得某个list/array/对象里面的字段集合

    java8新特性: lambda表达式:直接获得某个list/array/对象里面的字段集合 比如,我有一张表: entity Category.java service CategoryServic ...

  2. Java8 新特性学习 Lambda表达式 和 Stream 用法案例

    Java8 新特性学习 Lambda表达式 和 Stream 用法案例 学习参考文章: https://www.cnblogs.com/coprince/p/8692972.html 1.使用lamb ...

  3. Java8新特性之Lambda表达式

    lambda表达式是java8给我们带来的几个重量级新特性之一,借用lambda表达式,可以让我们的java程序设计更加简洁.最近新的项目摒弃了1.6的版本,全面基于java8进行开发,本文是java ...

  4. Java8 新特性之Lambda表达式

    1. Lambda 表达式概述 Lambda 是一个匿名函数,我们可以把 Lambda 表达式理解为是一段可以传递的代码(将代码像数据一样进行传递); Lambda 表达式可以写出更简洁,更灵活的代码 ...

  5. 【Java8新特性】Lambda表达式基础语法,都在这儿了!!

    写在前面 前面积极响应读者的需求,写了两篇Java新特性的文章.有小伙伴留言说:感觉Lambda表达式很强大啊!一行代码就能够搞定那么多功能!我想学习下Lambda表达式的语法,可以吗?我的回答是:没 ...

  6. 【Java8新特性】- Lambda表达式

    Java8新特性 - Lambda表达式 生命不息,写作不止 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 一个有梦有戏的人 @怒放吧德德 分享学习心得,欢迎指正,大家一起学习成长! ...

  7. 夯实Java基础(二十二)——Java8新特性之Lambda表达式

    1.前言 Java 8于14年发布到现在已经有5年时间了,经过时间的磨练,毫无疑问,Java 8是继Java 5(发布于2004年)之后的又一个非常最重要的版本.因为Java 8里面出现了非常多新的特 ...

  8. java8新特性之——lambda表达式的使用

    lambda表达式简介 个人理解,lambda表达式就是一种新的语法,没有什么新奇的,简化了开发者的编码,其实底层还是一些常规的代码.Lambda 是一个匿名函数,我们可以把 Lambda 表达式理解 ...

  9. 【Java8新特性】Lambda表达式

    一.Lambda 表达式 是什么? Lambda读音:拉姆达. Lambda是一个匿名函数,匿名函数就是一个没有名字的函数. Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中). ...

随机推荐

  1. pytorch lstm crf 代码理解 重点

    好久没有写博客了,这一次就将最近看的pytorch 教程中的lstm+crf的一些心得与困惑记录下来. 原文 PyTorch Tutorials 参考了很多其他大神的博客,https://blog.c ...

  2. 自动为DEV GridView控件添加SizeChanged事件

    实现gdv设置的抽象对象,不用每个gdv控件都添加sizechanged事件,只需执行gdc绑定sql函数,在其中会自动添加SizeChanged事件. //2016.5.13 by sngk //根 ...

  3. Group_concat介绍与例子

    进公司做的第一个项目就是做一个订单追踪查询,里里外外连接了十一个表,作为公司菜鸡的我麻了爪. 其中有一个需求就是对于多行的数据在一行显示,原谅我才疏学浅 无奈下找到了项目组长  在那学来了这个利器 ( ...

  4. Postman接口测试工具学习笔记

    - 新建测试接口 在Postman中有两种新建测试接口的方式,第一种是图片右上角的,点击可以选择 request 请求进行新建 选择 Request 以后会出现下面图片的对话框,让你输入一个保存接口的 ...

  5. JAVA配置系统变量

    CLASSPATH= .;%JAVA_HOME%/lib/dt.jar;%JAVA_HOME%/lib/tools.jar JAVA_HOME = C:/Program Files/Java/jdk1 ...

  6. java基本数据类型和包装类相互转换

    把基本数据类型 → 包装类: 通过对应包装类的构造方法实现 除了Character外,其他包装类都可以传入一个字符串参数构建包装类对象. 包装类 → 基本数据类型 包装类的实例方法xxxValue() ...

  7. window 系统下修改`CMD`的编码格式的方法,`CHCP` 的 使用

    CHCP的使用 CHCP是一个计算机指令,能够显示或设置活动代码页编号. 一般上是在命令提示框中使用,用来查询和修改命令提示框的编码格式 具体使用方法 查看活动代码页编号 方式1: >>& ...

  8. 2018-2-13-WPF-拖动时出现-Invalid-FORMATETC-structure

    title author date CreateTime categories WPF 拖动时出现 Invalid FORMATETC structure lindexi 2018-2-13 17:2 ...

  9. windows常用命令行命令

    https://blog.csdn.net/qq_32451373/article/details/77743869 打开"运行"对话框(Win+R),输入cmd,打开控制台命令窗 ...

  10. P1059 硬币翻转

    题目描述 从前有很多个硬币摆在一行,有正面朝上的,也有背面朝上的.正面朝上的用1表示,背面朝上的用0表示.现在要求从这行的第一个硬币开始,将前若干个硬币一起翻面,问如果要将所有硬币翻到正面朝上,最少要 ...