Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >Lamda 表达式作用域和内置函数式接口

Lamda 表达式作用域和内置函数式接口

作者头像
happyJared
发布于 2019-07-31 09:49:31
发布于 2019-07-31 09:49:31
56200
代码可运行
举报
文章被收录于专栏:happyJaredhappyJared
运行总次数:0
代码可运行

Lamda 表达式作用域

访问局部变量

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

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
final int num = 1;
Converter<Integer, String> stringConverter = (from) -> String.valueOf(from + num);
stringConverter.convert(2);  // 3

但是和匿名对象不同的是,这里的变量 num 可以不用声明为 final,该代码同样正确:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
int num = 1;
Converter<Integer, String> stringConverter = (from) -> String.valueOf(from + num);
stringConverter.convert(2);     // 3

不过这里的 num 必须不可被后面的代码修改(即隐性的含有 final 语义),例如下面的就无法编译:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
int num = 1;
Converter<Integer, String> stringConverter = (from) -> String.valueOf(from + num);
num = 3;  // 在 lambda 表达式中试图修改 num 同样是不允许的
访问字段和静态变量

与局部变量相比,对 lambda 表达式中的实例字段和静态变量都有读写访问权限,该行为和匿名对象是一致的:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class Lambda4 {
    static int outerStaticNum;
    int outerNum;

    void testScopes() {
        Converter<Integer, String> stringConverter1 = (from) -> {
            outerNum = 23;
            return String.valueOf(from);
        };

        Converter<Integer, String> stringConverter2 = (from) -> {
            outerStaticNum = 72;
            return String.valueOf(from);
        };
    }
}
访问默认接口方法

还记得上一篇文章中的 formula 示例吗? Formula 接口定义了一个默认方法 sqrt(),可以从包含匿名对象的每个 formula 实例访问该方法,不过这不适用于 lambda 表达式,无法从 lambda 表达式中访问默认方法,以下代码无法通过编译:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Formula formula = (a) -> sqrt(a * 100);

内置函数式接口

JDK1.8 API 包含许多内置函数式接口。 其中一些借口在老版本的 Java 中是比较常见的,比如: ComparatorRunnable,这些接口都增加了 @FunctionalInterface 注解,以便能用在 lambda 表达式上。

但是 Java8 API 同样还提供了很多全新的函数式接口来让编程工作更加方便,有一些接口是来自 Google Guava 库里的,来看看这些是如何扩展到 lambda 上使用的。

Predicates

Predicate 接口是只有一个参数的返回布尔类型值的 断言型 接口。该接口包含多种默认方法来将 Predicate 组合成其他复杂的逻辑(比如:与,或,非):

Predicate 接口源码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package java.util.function;
import java.util.Objects;

@FunctionalInterface
public interface Predicate<T> {

    // 该方法是接受一个传入类型,返回一个布尔值.此方法应用于判断
    boolean test(T t);

    // and 方法与关系型运算符"&&"相似,两边都成立才返回 true
    default Predicate<T> and(Predicate<? super T> other) {
        Objects.requireNonNull(other);
        return (t) -> test(t) && other.test(t);
    }
    // 与关系运算符"!"相似,对判断进行取反
    default Predicate<T> negate() {
        return (t) -> !test(t);
    }
    // or 方法与关系型运算符"||"相似,两边只要有一个成立就返回 true
    default Predicate<T> or(Predicate<? super T> other) {
        Objects.requireNonNull(other);
        return (t) -> test(t) || other.test(t);
    }
   // 该方法接收一个 Object 对象,返回一个 Predicate 类型。此方法用于判断第一个 test 的方法与第二个 test 方法相同(equal)
    static <T> Predicate<T> isEqual(Object targetRef) {
        return (null == targetRef)
                ? Objects::isNull
                : object -> targetRef.equals(object);
    }

示例:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Predicate<String> predicate = (s) -> s.length() > 0;

predicate.test("foo");              // true
predicate.negate().test("foo");     // false

Predicate<Boolean> nonNull = Objects::nonNull;
Predicate<Boolean> isNull = Objects::isNull;

Predicate<String> isEmpty = String::isEmpty;
Predicate<String> isNotEmpty = isEmpty.negate();
Functions

Function 接口接受一个参数并生成结果。默认方法可用于将多个函数链接在一起(compose, andThen):

Function 接口源码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package java.util.function;

import java.util.Objects;

@FunctionalInterface
public interface Function<T, R> {

    // 将Function对象应用到输入的参数上,然后返回计算结果
    R apply(T t);

    // 将两个Function整合,并返回一个能够执行两个 Function 对象功能的 Function 对象
    default <V> Function<V, R> compose(Function<? super V, ? extends T> before) {
        Objects.requireNonNull(before);
        return (V v) -> apply(before.apply(v));
    }

    default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
        Objects.requireNonNull(after);
        return (T t) -> after.apply(apply(t));
    }

    static <T> Function<T, T> identity() {
        return t -> t;
    }
}
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Function<String, Integer> toInteger = Integer::valueOf;
Function<String, String> backToString = toInteger.andThen(String::valueOf);
backToString.apply("123");  // "123"
Suppliers

Supplier 接口产生给定泛型类型的结果。 与 Function 接口不同,Supplier 接口不接受参数。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Supplier<Person> personSupplier = Person::new;
personSupplier.get();  // new Person
Consumers

Consumer 接口表示要对单个输入参数执行的操作。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Consumer<Person> greeter = (p) -> System.out.println("Hello, " + p.firstName);
greeter.accept(new Person("Luke", "Skywalker"));
Comparators

Comparator 是老版本中的经典接口, Java8 在此之上添加了多种默认方法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Comparator<Person> comparator = (p1, p2) -> p1.firstName.compareTo(p2.firstName);

Person p1 = new Person("John", "Doe");
Person p2 = new Person("Alice", "Wonderland");

comparator.compare(p1, p2);             // > 0
comparator.reversed().compare(p1, p2);  // < 0
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2019.07.30 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Lamda 表达式作用域(Lambda Scopes)
但是和匿名对象不同的是,这里的变量num可以不用声明为final,该代码同样正确:
崔笑颜
2020/06/08
5550
函数式接口
所谓的函数式接口,实际上就是接口里面只能有一个抽象方法的接口。Comparator接口就是一个典型的函数式接口,它只有一个抽象方法compare。
HLee
2021/08/15
7170
函数式接口
java8 新特性
Java 8 Tutorial 接口的默认方法(Default Methods for Interfaces) Lambda表达式(Lambda expressions) 函数式接口(Functional Interfaces) 方法和构造函数引用(Method and Constructor References) Lamda 表达式作用域(Lambda Scopes) 访问局部变量 访问字段和静态变量 访问默认接口方法 内置函数式接口(Built-in Functional Inter
袁新栋-jeff.yuan
2020/08/26
5260
深入探寻JAVA8 part2:浅谈几个内置的函数式接口
函数式接口就是只定义一个抽象方法的接口。在JAVA8以前,就有很多符合函数式接口定义的接口。
眯眯眼的猫头鹰
2019/10/15
5200
玩转Java8的 Stream 之函数式接口
函数式接口是伴随着Stream的诞生而出现的,Java8Stream 作为函数式编程的一种具体实现,开发者无需关注怎么做,只需知道要做什么,各种操作符配合简洁明了的函数式接口给开发者带来了简单快速处理数据的体验。
搜云库技术团队
2020/06/04
4090
Java的函数式接口以及Lambda表达式
在java中,大家肯定使用过lambda表达式吧,这是适用于函数式接口的一种便捷写法。
半月无霜
2023/03/03
4540
Java的函数式接口以及Lambda表达式
【小家java】java8新特性之---函数式接口(Supplier、Consumer、Predicate、Function、UnaryOperator,通往高阶设计的好工具)
首先,它还是一个接口,所以必须满足接口最基本的定义。但它是一个特殊的接口:SAM类型的接口(Single Abstract Method)。可以在调用时,使用一个lambda表达式作为参数。 定义要求:
YourBatman
2019/09/03
1K0
【小家java】java8新特性之---函数式接口(Supplier、Consumer、Predicate、Function、UnaryOperator,通往高阶设计的好工具)
【Java 8】Lambda 表达式
Lambda(λ) 表达式是一种在 被调用的位置 或者 作为参数传递给函数的位置 定义匿名函数对象 的简便方法。下面是关于 Lambda 表达式的几个点:
零式的天空
2022/03/27
6320
JDK1.8新特性(四):函数式接口
上一篇JDK1.8新特性(三):Lambda表达式,让你爱不释手,只是简单的讲到Lambda表达式的语法、使用,使得你对它产生了好感,而Lambda表达式是如何实现、定义,你可能不太清楚。本篇将会详细介绍函数式接口,让你在使用JDK新特性时,做到心中有数,自信满满。
xcbeyond
2020/05/31
1.1K0
JDK1.8新特性(四):函数式接口
Java 8 函数式接口使用示例及相关新特性
函数式接口(Functional Interface)是Java 8对一类特殊类型的接口的称呼。 函数式接口下共有
chenchenchen
2020/05/26
1.7K0
Java8新增方法使用
Java8允许我们使用default关键字为接口添加非抽象的方法。这个特点也被称为扩展方法,下面是例子:
小森啦啦啦
2019/12/15
1.5K0
JAVA8十大新特性详解
Java 8允许我们给接口添加一个非抽象的方法实现,只需要使用 default关键字即可,这个特征又叫做扩展方法,示例如下:
用户3467126
2019/11/01
9400
Java 8简明教程
ImportNew注:有兴趣第一时间学习Java 8的Java开发者,欢迎围观《征集参与Java 8原创系列文章作者》。 以下是《Java 8简明教程》的正文。 “Java并没有没落,人们很快就会发现这一点” 欢迎阅读我编写的Java 8介绍。本教程将带领你一步一步地认识这门语言的新特性。通过简单明了的代码示例,你将会学习到如何使用默认接口方法,Lambda表达式,方法引用和重复注解。看完这篇教程后,你还将对最新推出的API有一定的了解,例如:流控制,函数式接口,map扩展和新的时间日期API等等。 允许在
美的让人心动
2018/06/14
1.1K0
干货 | Java8 新特性指导手册
小伙伴们,Java 版本都到 12 啦,我发现还有很多小伙伴对 8 的新特性不是很了解。
Java技术江湖
2019/09/24
1.4K0
干货 | Java8 新特性指导手册
拉姆达表达式是什么_拉姆达
通常我们会用一个类实现接口,然后构造对象作为参数传入,也可以使用匿名类,用λ表达式可以简化匿名类的编写,用例如下。
全栈程序员站长
2022/11/04
3410
拉姆达表达式是什么_拉姆达
生产上常用 JDK 版本 1.7 与 1.8 新特性
JDK7开始,终于可以用二进制来表示整数(byte,short,int和long)。使用二进制字面量的好处是,可以是代码更容易被理解。语法非常简单,只要在二进制数值前面加 0b或者0B
BUG弄潮儿
2021/12/01
1.2K0
隔壁老王都熟练使用函数式接口了,你还在等什么?(快来收藏)
  最近刚好有空给大家整理下JDK8的特性,这个在实际开发中的作用也是越来越重了,本文重点讲解下函数式接口内容。
用户4919348
2021/06/10
2560
函数式接口
发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/100187.html原文链接:
全栈程序员站长
2021/05/20
5040
Lambda01-Functional 函数式接口
Functional 的定义其实很简单:任何接口,如果只包含 唯一一个抽象方法,那么它就是一个 Functional Interface , 函数式接口?这注解名字好。 为了让编译器帮助我们确保一个接口满足 Functional Interface 的要求(也就是说有且仅有一个抽象方法),Java8提供了@FunctionalInterface注解。
潇洒
2023/10/20
1510
java函数式接口
这个接口可以使用@FunctionalInterface声明,如果不满足条件,则会报错,可以通过这个方式来判断是否为函数式接口
仙士可
2023/03/09
3880
java函数式接口
相关推荐
Lamda 表达式作用域(Lambda Scopes)
更多 >
领券
社区富文本编辑器全新改版!诚邀体验~
全新交互,全新视觉,新增快捷键、悬浮工具栏、高亮块等功能并同时优化现有功能,全面提升创作效率和体验
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验