Java8学习笔记之其他语言特性

2019-08-14  本文已影响0人  夏与清风

Java 8中包含三个语言特性的更新,分别是重复注解(repeated annotation)、类型注解(type annotation)和通用目标类型推断(generalized target-type inference)。

1、注解

Java 8在两个方面对注解机制进行了改进:

Java中的注解是一种对程序元素进行配置,提供附加信息的机制(在Java 8之前,只有声明可以被注解)。它是某种形式的语法元数据(syntactic metadata)。比如,注解在JUnit框架中就使用得非常频繁。

@Before
public void setUp(){
  this.list = new ArrayList<>();
}
 
@Test
public void testAlgorithm(){
  ...
  assertEquals(5, list.size());
}

注解尤其适用于下面这些场景:

1)重复注解

之前版本的Java禁止对同样的注解类型声明多次。

@interface
Author {String name();}
 
@Author(name="Raoul") @Author(name="Mario") @Author(name="Alan") //错误:重复的注解
class Book{}  

程序员经常通过一些惯用法绕过这一限制。你可以声明一个新的注解,它包含了希望重复的注解数组。这种方法的形式如下:

@interface
Author {String name();}
@interface
Authors {Author[] value();}

@Authors(
  {@Author(name="Raoul"),@Author(name="Mario"),@Author(name="Alan")}
)
class Book{}

Book类的嵌套注解相当难看。这就是Java 8要从根本上移除这一限制的原因,去掉这一限制后,代码的可读性会好很多。现在,如果你的配置允许重复注解,你可以毫无顾虑地一次声明多个同一种类型的注解。它目前还不是默认行为,需要显式地要求进行重复注解。
如果一个注解在设计之初就是可重复的,你可以直接使用它。但是,如果提供的注解是为用户提供的,那么就需要做一些工作,说明该注解可以重复。下面是你需要执行的两个步骤:

@Repeatable(Authors.class)
@interface
Author {
  String name();
}
@interface Authors {
  Author[] value();
}

@Author(name="Raoul") @Author(name="Mario") @Author(name="Alan")
class Book{}

编译时,Book会被认为使用了@Authors({@Author(name="Raoul"), @Author(name =”Mario”), @Author(name=”Alan”)})这样的形式进行了注解。可以把这种新的机制看成是一种语法糖,它提供了Java程序员之前利用的惯用法类似的功能。为了确保与反射方法在行为上的一致性,注解会被封装到一个容器中。
Java API中的getAnnotation(Class<T> annotation-Class)方法会为注解元素返回类型为T的注解。如果实际情况有多个类型为T的注解,该方法的返回到底是哪一个呢?
类Class提供了一个新的getAnnotationsByType方法,它可以帮助我们更好地使用重复注解。比如,可以像下面这样打印输出Book类的所有 Author注解:

public static void main(String[] args) {
  Author[] authors = Book.class.getAnnotationsByType(Author.class); //返回一个由重复注解Author组成的数组
  Arrays.asList(authors)
    .forEach(a -> {System.out.println(a.name());});
}
2)类型注解

从Java 8开始,注解已经能应用于任何类型。包括new操作符、类型转换、instanceof 检查、泛型类型参数,以及implements和throws子句。

@NonNull
String name = person.getName();

List<@NonNull Car> cars = new ArrayList<>();

Java 8并未提供官方的注解或者一种工具能以开箱即用的方式使用它们。它仅仅提供了一种功能,使用它可以对不同的类型添加注解。

2、通用目标类型推断

Java 8对泛型参数的推断进行了增强。比如,Java中的方法emptyList方法定义如下:

static <T> List<T> emptyList();

emptyList方法使用了类型参数T进行参数化。可以像下面这样为该类型参数提供一个显式的类型进行函数调用:

List<Car> cars = Collections.<Car>emptyList();

Java也可以推断泛型参数的类型。上面的代码和下面这段代码是等价的:

List<Car> cars = Collections.emptyList();

Java 8之前,这种推断机制依赖于程序的上下文(即目标类型),具有一定的局限性。比如下面这种情况就不大可能完成推断:

static void cleanCars(List<Car> cars) {}
cleanCars(Collections.emptyList());

会遭遇下面的错误:

cleanCars (java.util.List<Car>)cannot be applied to (java.util.List<java.lang.Object>)

为了修复此问题,只能提供一个显式的类型参数。Java 8中,目标类型包括向方法传递的参数,不再需要提供显式的泛型参数:

List<Car> cleanCars = dirtyCars.stream()
    .filter(Car::isClean)
    .collect(Collectors.toList());

Java 8的改进只需一句Collectors.toList()就能完成期望的工作,不再需要编写像Collectors.<Car>toList()这么复杂的代码了。

--参考文献《Java8实战》

上一篇 下一篇

猜你喜欢

热点阅读