Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >Java 8 - 04 类型检查、类型推断以及限制

Java 8 - 04 类型检查、类型推断以及限制

作者头像
小小工匠
发布于 2021-08-17 03:34:22
发布于 2021-08-17 03:34:22
88600
代码可运行
举报
文章被收录于专栏:小工匠聊架构小工匠聊架构
运行总次数:0
代码可运行

Pre

当我们第一次提到Lambda表达式时,说它可以为函数式接口生成一个实例。然而,Lambda 表达式本身并不包含它在实现哪个函数式接口的信息。为了全面了解Lambda表达式,women 应该知道Lambda的实际类型是什么 .


类型检查

Lambda的类型是从使用Lambda的上下文推断出来的。 上下文(比如,接受它传递的方法的参数,或接受它的值的局部变量)中Lambda表达式需要的类型称为目标类型。

举个例子

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
List<Apple> heavierThan150g = filter(inventory, (Apple a) -> a.getWeight() > 150);

类型检查过程可以分解为如下所示。  首先,我们要找出 filter 方法的声明。  第二,要求它是 Predicate (目标类型)对象的第二个正式参数。  第三, Predicate 是一个函数式接口,定义了一个叫作 test 的抽象方法。  第四, test 方法描述了一个函数描述符,它可以接受一个 Apple ,并返回一个 boolean 。  最后, filter 的任何实际参数都必须匹配这个要求

这段代码是有效的,因为我们所传递的Lambda表达式也同样接受 Apple 为参数,并返回一个boolean 。请注意,如果Lambda表达式抛出一个异常,那么抽象方法所声明的 throws 语句也必须与之匹配


同样的 Lambda,不同的函数式接口

有了目标类型的概念,同一个Lambda表达式就可以与不同的函数式接口联系起来,只要它们的抽象方法签名能够兼容.

我们来看下这两个函数式接口

这两个函数式接口 都是 什么也不接受且返回一个泛型 T 的函数, 所以 下面两个赋值是有效的

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 Callable<Integer> integerCallable = () -> 18;
 PrivilegedAction<Integer> privilegedAction = () -> 18;

第一个赋值的目标类型是 Callable 第二个赋值的目标类型是PrivilegedAction

再举个栗子 : 同一个Lambda可用于多个不同的函数式接口

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    Comparator<Enginner> enginnerComparator = (e1, e2) -> e1.getJob().compareTo(e2.getJob());
    ToIntBiFunction<Enginner, Enginner> toIntBiFunction = (e1, e2) -> e1.getJob().compareTo(e2.getJob());
    BiFunction<Enginner, Enginner, Integer> toIntFunction = (e1, e2) -> e1.getJob().compareTo(e2.getJob());

Comparator 、 ToIntBiFunction 、 BiFunction 都是返回一个int类型的的函数


菱形运算符

Java 7中已经引入了菱形运算符( <> ),利用泛型推断从上下文推断类型的思想。 一个类实例表达式可以出现在两个或更多不同的上下文中,并会像下面这样推断出适当的类型参数。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
List<String> listOfStrings = new ArrayList<>();
List<Integer> listOfIntegers = new ArrayList<>();

特殊的void兼容规则

如果一个Lambda的主体是一个语句表达式, 它就和一个返回 void 的函数描述符兼容(当然需要参数列表也兼容)。

举个例子:

以下两行都是合法的,尽管 List 的 add 方法返回了一个boolean ,而不是 Consumer 上下文( T -> void )所要求的 void

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
   List<String> stringList = new ArrayList<>();
  // Predicate返回了一个boolean
  Predicate<String> predicate = s -> stringList.add(s);
   // Consumer返回了一个void
   Consumer<String> consumer =    s -> stringList.add(s);

经过了这几个小demo ,是不是能够很好地理解在什么时候以及在哪里可以使用Lambda表达式了。Lambda表达式可以从赋值的上下文、方法调用的上下文(参数和返回值),以及类型转换的上下文中获得目标类型

来个小测验

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
类型检查——为什么下面的代码不能编译呢?
Object o = () -> {System.out.println("Tricky example"); };

答案: 
Lambda表达式的上下文是 Object (目标类型)。但 Object 不是一个函数式接口 。 为了解决这个问题,可以把目标类型改成 Runnable ,它的函数描述符是 () -> void :

Runnable r = () -> {System.out.println("Tricky example"); };

类型推断

刚才已经讨论了如何利用目标类型来检查一个Lambda是否可以用于某个特定的上下文。其实, 它也可以用来做一些略有不同的事:推断Lambda参数的类型,我们来看下。

Java编译器会从上下文(目标类型)推断出用什么函数式接口来配合Lambda表达式,这意味着它也可以推断出适合Lambda的签名,因为函数描述符可以通过目标类型来得到。这样做的好处在于,编译器可以了解Lambda表达式的参数类型,这样就可以在Lambda语法中省去标注参数类型.

举个例子

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 List<Enginner> goEngineerList = filter(enginnerList,a-> a.getJob().equals("GO"));

参数 a 没有显式类型 .

再举个栗子 ,Lambda表达式有多个参数,代码可读性的好处就更为明显

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
     // 没有类型推断,因为给o1,o2指定了Enginner 类型
    Comparator<Enginner> comparator = (Enginner o1, Enginner o2) -> o1.getJob().compareTo(o2.getJob());
    

     //  有类型推断,因为没有给o1,o2指定了Enginner 类型
     Comparator<Enginner> comparator2 = ( o1,  o2) -> o1.getJob().compareTo(o2.getJob());

个人感觉,第二种写法更简单 。

当Lambda仅有一个类型需要推断的参数时,参数名称两边的括号也可以省略。


使用局部变量

上面所介绍的所有Lambda表达式都只用到了其主体里面的参数。但Lambda表达式也允许使用自由变量(不是参数,而是在外层作用域中定义的变量),就像匿名类一样。 它们被称作捕获Lambda。

举个例子

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  int num = 1;
  Runnable runnable = ()->System.out.println(num);

这么做虽然有点啰嗦,我们这里想要讨论的是 使用外部的变量有什么限制吗?

如果你想要对这个变量进行操作,之前的lambda就报错了。所以说Lambda可以没有限制地捕获(也就是在其主体中引用)实例变量和静态变量,但是局部变量必须显式声明为 final.

换句话说,Lambda表达式只能捕获指派给它们的局部变量一次。(注:捕获实例变量可以被看作捕获最终局部变量 this 。) 如上图。

为什么会这样呢?

  • 第一: 实例变量都存储在堆中,而局部变量则保存在栈上。如果Lambda可以直接访问局部变量,而且Lambda是在一个线程中使用的,则使用Lambda的线程,可能会在分配该变量的线程将这个变量收回之后,去访问该变量。因此,Java在访问自由局部变量时,实际上是在访问的副本,而不是访问原始变量。如果局部变量仅仅赋值一次那就没有什么区别了——因此就有了这个限制
  • 第二,这一限制不鼓励你使用改变外部变量的典型命令式编程模式,这种模式会阻碍很容易做到的并行处理.
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2020/05/16 ,如有侵权请联系 cloudcommunity@tencent.com 删除

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

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

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

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Java 8 - 02 Lambda Expression
上一节 Java 8 - 01 优雅编程 lambda 以及 @FunctionalInterface注解一点通 中有的时候使用了匿名类来表示不同的行为 ,代码比较啰嗦
小小工匠
2021/08/17
4720
Java 8 - 06 Lambda 和方法引用实战
前几篇文章,我们已经学习了行为参数化、匿名类、Lambda表达式和方法引用,实现了给工程师按照职位排序的功能
小小工匠
2021/08/17
3350
Java 8 - 05 方法引用
方法引用让你可以重复使用现有的方法定义,并像Lambda一样传递它们。在一些情况下比起使用Lambda表达式, 更易读 。上面的栗子就是借助了Java 8 API ,用方法引用写的一个排序的例子。
小小工匠
2021/08/17
4720
Java8新特性----Lambda表达式详细探讨
​ Lambda是一个匿名函数,可以理解为一段可以传递的代码(将代码像数据一样传递);可以写出更简洁、更灵活的代码;作为一种更紧凑的代码风格,是Java语言表达能力得到提升
大忽悠爱学习
2021/11/15
2800
【JDK1.8 新特性】Lambda表达式
Lambda 是一个匿名函数,我们可以把 Lambda 表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。使用它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升。
CODER-V
2023/03/08
2580
Java8学习(3)- Lambda 表达式
猪脚:以下内容参考《Java 8 in Action》 本次学习内容: Lambda 基本模式 环绕执行模式 函数式接口,类型推断 方法引用 Lambda 复合 代码: https://github.com/Ryan-Miao/someTest/blob/master/src/main/java/com/test/java8/c3/AppleSort.java 上一篇: Java8学习(2)- 通过行为参数化传递代码--lambda代替策略模式 ---- 1. 结构 初始化一个比较器: Comparato
Ryan-Miao
2018/03/13
1K0
Java8学习(3)- Lambda 表达式
解构Lambda表达式
把方法作为值传递虽然很有用,但是有时有些方法我们不想具体定义,这时候Lambda函数就登场了。Lambda表达式的基本语法是(parameters)->expression或者(parameters)->{statements;}
简熵
2023/03/06
3010
解构Lambda表达式
Java8之熟透Lambda表达式
​ Lambda 表达式可以理解为简洁地表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表、函数主体、返回类型,可能还有一个可以抛出的异常列表。
用户1195962
2019/09/29
5670
Java8之熟透Lambda表达式
java8实战读书笔记:Lambda表达式语法与函数式编程接口
测试:如下语句是否是正确的lambda表达式。 (1) () -> {} (2) () -> "Raoul" (3) () -> {return "Mario";} (4) (Integer i) -> return "Alan" + i; (5) (String s) -> {"IronMan";}
丁威
2019/06/11
5290
java8实战读书笔记:Lambda表达式语法与函数式编程接口
系统学习Lambda表达式
在《挑苹果中的行为参数化思想》已经介绍了用Lambda表达式将行为抽象化,对Lambda表达式有一定认识。而本文将对Lambda表达式进行系统性的介绍。
草捏子
2020/08/10
6130
系统学习Lambda表达式
Java8新特性——Lambda表示式
👨‍🎓作者:Java学术趴 🏦仓库:Github、Gitee ✏️博客:CSDN、掘金、InfoQ、云+社区 💌公众号:Java学术趴 🚫特别声明:原创不易,未经授权不得转载或抄袭,如需转载可联系小编授权。 🙏版权声明:文章里的部分文字或者图片来自于互联网以及百度百科,如有侵权请尽快联系小编。 ☠️每日毒鸡汤:放心,闭上眼,睡一觉,反正明天也不一定比今天好。 1.1 Java8的新特性 1.2 Java8新特性的特点 速度快。 代码更少(增加了新的语法:Lambda表达式)
Java学术趴
2022/08/09
6100
Java8新特性——Lambda表示式
【Java8新特性】Lambda表达式基础语法,都在这儿了!!
Lambda表达式在Java语言中引入了 “->” 操作符, “->” 操作符被称为Lambda表达式的操作符或者箭头操作符,它将Lambda表达式分为两部分:
冰河
2020/10/29
3520
【Java8新特性】Lambda表达式基础语法,都在这儿了!!
死磕Lambda表达式(三):更简洁的Lambda
在之前的文章中介绍了Lambda表达式的基本语法和正确使用姿势,这次我来介绍一些Lambda更简洁的用法。
万猫学社
2022/04/22
2260
死磕Lambda表达式(三):更简洁的Lambda
Java 8 - 07 复合 Lambda 表达式
事实上,许多函数式接口,比如用于传递Lambda表达式的 Comparator 、 Function 和 Predicate 都提供了允许你进行复合的方法
小小工匠
2021/08/17
4620
Java8新特性详解
Java 8 是oracle公司于2014年3月发布,可以看成是自Java 5 以来最具革命性的版本。Java 8为Java语言、编译器、类库、开发工具与JVM带来了大量新特性。
Jensen_97
2023/07/20
2.2K0
Java8新特性详解
Java函数式编程和Lambda表达式
相信大家都使用过面向对象的编程语言,面向对象编程是对数据进 行抽象,而函数式编程是对行为进行抽象。函数式编程让程序员能够写出更加容易阅读的代码。那什么时候函数式编程呢?
程序那些事
2020/07/08
7190
一文看懂 Java8 的 Lambda表达式!
IT领域的技术日新月异,Java14问世了,但是对于国内的许多程序员来说,连Java8都还没有真正掌握。
程序员小跃
2020/04/10
4550
[二] java8 函数式接口详解 函数接口详解 lambda表达式 匿名函数 方法引用使用含义 函数式接口实例 如何定义函数式接口
        比如接收双参数的,有 Bi 前缀, 比如 BiConsumer<T,U>, BiFunction<T,U,R> ;
noteless
2018/09/11
1.8K0
[二] java8 函数式接口详解 函数接口详解 lambda表达式 匿名函数 方法引用使用含义 函数式接口实例  如何定义函数式接口
java8新特性(一):Lambda表达式
Lambda 是一个匿名函数,我们可以把Lambda表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升。
周三不加班
2019/09/03
3980
java8新特性(一):Lambda表达式
Java 8——Lambda表达式
本文内容大部分来自《Java 8实战》一书 前言 在上一篇文章中,我们了解了利用行为参数化来传递代码有助于应对不断变化的需求,它允许你定义一个代码块来表示一个行为,然后传递它。一般来说,利用这个概念,你就可以编写更为灵活且可重复使用的代码了。 但是你同时也看到,使用匿名类来表示不同的行为并不令人满意:代码十分啰嗦,这会影响程序员在时间中使用行为参数化的积极性。Lambda表达式很好的解决了这个问题,它可以让你很简洁地表示一个行为或传递代码。现在你可以把Lambda表达式看作匿名功能,它基本上就是没有声明名
我没有三颗心脏
2018/04/26
1.1K0
Java 8——Lambda表达式
相关推荐
Java 8 - 02 Lambda Expression
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验