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

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

20200510181139786.png


Pre


当我们第一次提到Lambda表达式时,说它可以为函数式接口生成一个实例。然而,Lambda

表达式本身并不包含它在实现哪个函数式接口的信息。为了全面了解Lambda表达式,women 应该知道Lambda的实际类型是什么 .


类型检查


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


举个例子

List<Apple> heavierThan150g = filter(inventory, (Apple a) -> a.getWeight() > 150);



20200516232618286.png

类型检查过程可以分解为如下所示。

 首先,我们要找出 filter 方法的声明。

 第二,要求它是 Predicate<Apple> (目标类型)对象的第二个正式参数。

 第三, Predicate<Apple> 是一个函数式接口,定义了一个叫作 test 的抽象方法。

 第四, test 方法描述了一个函数描述符,它可以接受一个 Apple ,并返回一个 boolean 。

 最后, filter 的任何实际参数都必须匹配这个要求


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


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


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

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


20200517071907938.png


20200517071942571.png


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

 Callable<Integer> integerCallable = () -> 18;
 PrivilegedAction<Integer> privilegedAction = () -> 18;

第一个赋值的目标类型是 Callable<Integer>

第二个赋值的目标类型是PrivilegedAction<Integer>


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

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

List<String> listOfStrings = new ArrayList<>();
List<Integer> listOfIntegers = new ArrayList<>();


特殊的void兼容规则


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

举个例子:


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

   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表达式可以从赋值的上下文、方法调用的上下文(参数和返回值),以及类型转换的上下文中获得目标类型

来个小测验

类型检查——为什么下面的代码不能编译呢?
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语法中省去标注参数类型.


举个例子


List<Enginner> goEngineerList = filter(enginnerList,a-> a.getJob().equals("GO"));


参数 a 没有显式类型 .

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

     // 没有类型推断,因为给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。

举个例子

  int num = 1;
  Runnable runnable = ()->System.out.println(num);


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

20200517083441248.png


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


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


为什么会这样呢?


第一: 实例变量都存储在堆中,而局部变量则保存在栈上。如果Lambda可以直接访问局部变量,而且Lambda是在一个线程中使用的,则使用Lambda的线程,可能会在分配该变量的线程将这个变量收回之后,去访问该变量。因此,Java在访问自由局部变量时,实际上是在访问的副本,而不是访问原始变量。如果局部变量仅仅赋值一次那就没有什么区别了——因此就有了这个限制

第二,这一限制不鼓励你使用改变外部变量的典型命令式编程模式,这种模式会阻碍很容易做到的并行处理.

相关文章
|
22天前
|
安全 Java 编译器
Java类型提升与类型转换详解
本文详解Java中的类型提升与类型转换机制,涵盖类型提升规则、自动类型转换(隐式转换)和强制类型转换(显式转换)的使用场景与注意事项。内容包括类型提升在表达式运算中的作用、自动转换的类型兼容性规则,以及强制转换可能引发的数据丢失和运行时错误。同时提供多个代码示例,帮助理解byte、short、char等类型在运算时的自动提升行为,以及浮点数和整型之间的转换技巧。最后总结了类型转换的最佳实践,如避免不必要的转换、使用显式转换提高可读性、金融计算中使用BigDecimal等,帮助开发者写出更安全、高效的Java代码。
|
22天前
|
Java 测试技术
Java浮点类型详解:使用与区别
Java中的浮点类型主要包括float和double,它们在内存占用、精度范围和使用场景上有显著差异。float占用4字节,提供约6-7位有效数字;double占用8字节,提供约15-16位有效数字。float适合内存敏感或精度要求不高的场景,而double精度更高,是Java默认的浮点类型,推荐在大多数情况下使用。两者都存在精度限制,不能用于需要精确计算的金融领域。比较浮点数时应使用误差范围或BigDecimal类。科学计算和工程计算通常使用double,而金融计算应使用BigDecimal。
299 6
|
19天前
|
安全 IDE Java
Java记录类型(Record):简化数据载体类
Java记录类型(Record):简化数据载体类
114 0
|
6月前
|
存储 传感器 缓存
java变量与数据类型:整型、浮点型与字符类型
### Java数据类型全景表简介 本文详细介绍了Java的基本数据类型和引用数据类型,涵盖每种类型的存储空间、默认值、取值范围及使用场景。特别强调了`byte`、`int`、`long`、`float`、`double`等基本类型在不同应用场景中的选择与优化,如文件流处理、金融计算等。引用数据类型部分则解析了`String`、数组、类对象、接口和枚举的内存分配机制。
169 15
|
10月前
|
存储 Java 开发者
Java 中 Set 类型的使用方法
【10月更文挑战第30天】Java中的`Set`类型提供了丰富的操作方法来处理不重复的元素集合,开发者可以根据具体的需求选择合适的`Set`实现类,并灵活运用各种方法来实现对集合的操作和处理。
587 113
|
5月前
|
Java
java中一个接口A,以及一个实现它的类B,一个A类型的引用对象作为一个方法的参数,这个参数的类型可以是B的类型吗?
本文探讨了面向对象编程中接口与实现类的关系,以及里氏替换原则(LSP)的应用。通过示例代码展示了如何利用多态性将实现类的对象传递给接口类型的参数,满足LSP的要求。LSP确保子类能无缝替换父类或接口,不改变程序行为。接口定义了行为规范,实现类遵循此规范,从而保证了多态性和代码的可维护性。总结来说,接口与实现类的关系天然符合LSP,体现了多态性的核心思想。
111 0
|
6月前
|
Java
课时11:Java数据类型划分(浮点类型)
课时11介绍了Java中的浮点数据类型。主要内容包括:1. 定义小数,默认使用Double类型;2. 定义Float变量,需在数值后加&quot;F&quot;或&quot;f&quot;进行强制转换;3. 观察不同类型计算结果,如Int型除法会丢失精度,需至少包含一个Double或Float类型以确保准确性。总结指出,在复杂计算中推荐使用Double类型以避免精度损失。
131 5
|
6月前
|
Java
课时10:Java数据类型划分(整型类型)
本文主要围绕Java中整型数据展开,详细讲解整型变量、常量的概念,整型数据运算规则,包括数据溢出问题及解决方法,数据类型转换(自动转换与强制转换)的原理和注意事项,同时介绍了整型数据默认值的相关知识,以及byte数据类型与int数据类型的关系和使用场景,帮助读者全面掌握Java整型数据的相关内容。
165 4
|
6月前
|
存储 Java 编译器
Java泛型类型擦除以及类型擦除带来的问题
本文主要讲解Java中的泛型擦除机制及其引发的问题与解决方法。泛型擦除是指编译期间,Java会将所有泛型信息替换为原始类型,并用限定类型替代类型变量。通过代码示例展示了泛型擦除后原始类型的保留、反射对泛型的破坏以及多态冲突等问题。同时分析了泛型类型不能是基本数据类型、静态方法中无法使用泛型参数等限制,并探讨了解决方案。这些内容对于理解Java泛型的工作原理和避免相关问题具有重要意义。
310 0
|
10月前
|
安全 Java 编译器
JDK 10中的局部变量类型推断:Java编程的简化与革新
JDK 10引入的局部变量类型推断通过`var`关键字简化了代码编写,提高了可读性。编译器根据初始化表达式自动推断变量类型,减少了冗长的类型声明。虽然带来了诸多优点,但也有一些限制,如只能用于局部变量声明,并需立即初始化。这一特性使Java更接近动态类型语言,增强了灵活性和易用性。
199 53