本篇文章列出了Java 8的重要功能,并附带了官方引入的示例。例如lambda表达式,Java流,功能接口和日期时间API等。Java SE 8于2014年初发布,在Java 8中,讨论最多的功能是lambda表达式。它还具有许多其他重要功能,例如接口的默认方法(Default Methods for Interfaces),流API和新的日期/时间API(Date API)。下面通过示例了解Java 8中的这些新功能。
目录
1. Lambda表达式(Lambda Expression)
Lambda表达式对于使用Scala等其他流行编程语言的我们很多人并不陌生。在Java编程语言中,Lambda表达式(或函数)只是一个匿名函数,即不带名称且不受标识符限制的函数。它们被准确地写在需要的地方,通常作为其他函数的参数。lambda表达式的基本语法为:
either
(parameters) -> expression
or
(parameters) -> { statements; }
or
一个典型的lambda表达式语法如下:
(x, y) -> x + y //This function takes two parameters and return their sum.
请注意,根据x和y的类型,方法可能会在多个地方使用。参数可以匹配int或Integer,也可以匹配String。
根据上下文,它将添加两个整数或连接两个字符串。
编写Lambda表达式的规则:
- Lambda表达式可以具有零个,一个或多个参数。
- 参数的类型可以显式声明,也可以从上下文中推断出来。
- 多个参数用强制括号括起来,并用逗号分隔。空括号用于表示一组空参数。
- 当存在单个参数时,如果推断出其类型,则不强制使用括号。例如a->返回a * a。
- Lambda表达式的主体可以包含零个,一个或多个语句。
- 如果lambda表达式的主体具有单个语句,则不必使用大括号,并且匿名函数的返回类型与主体表达式的返回类型相同。如果正文中的声明多于一个,则这些声明必须用大括号括起来。
2. 函数式接口(Functional Interface)
函数式接口也称为单一抽象方法接口(SAM接口)。顾名思义,它们允许其中仅包含一种抽象方法。Java 8引入了一个注解,即@FunctionalInterface当您注解的接口违反函数式接口的约定时,该注解会提示编译器级错误。
典型的函数式接口示例:
@FunctionalInterface
public interface MyFirstFunctionalInterface {
public void firstWork();
请注意,即使@FunctionalInterface省略注释,函数式接口也有效。它仅用于通知编译器在接口内部只能有一个抽象方法。另外,由于默认方法不是抽象的,因此可以随意向接口添加任意数量的默认方法。
要记住的另一个重要点是,如果接口声明的抽象方法重写了公共方法之一java.lang.Object,则该方法也不会计入接口的抽象方法计数,因为该接口的任何实现都是来自java.lang.Object或者其他地方。例如,下面是完全有效的函数式接口:
@FunctionalInterface
public interface MyFirstFunctionalInterface
{
public void firstWork();
@Override
public String toString(); //Overridden from Object class
@Override
public boolean equals(Object obj); //Overridden from Object class
}
3. 接口的默认方法(Default Methods)
Java 8允许您在接口中添加非抽象方法。必须将这些方法声明为默认方法。Java 8中引入了默认方法以启用lambda表达式的功能。
默认方法使您可以向库的接口添加新功能,并确保与为这些接口的较早版本编写的二进制代码兼容。
让我们看一个例子:
public interface Moveable {
default void move(){
System.out.println("I am moving");
}
}
Moveable接口定义了一种方法,move()并提供了默认实现。如果有任何类实现此接口,则无需实现其自身的move()方法版本。
它可以直接调用instance.move()。例如:
public class Animal implements Moveable{
public static void main(String[] args){
Animal tiger = new Animal();
tiger.move();
}
}
Output: I am moving
如果类想要使用自定义的move()方法,则它可以提供自己的自定义实现并覆盖该方法。
4. 流(Java 8 Streams)
Java 8 Streams API引入了另一个重大变化,该Java 8 Streams API提供了一种以各种方式处理一组数据的机制,
这些方式可以包括过滤,转换或对应用程序可能有用的任何其他方式。
Java 8中的Streams API支持不同类型的迭代,你可以在其中简单地定义要处理的集合,对每个item要执行的操作,
以及存储这些操作的输出。
下面是流API的示例。在此示例中,items是String类型的集合,要删除以某些前缀文本开头的条目。
List<String> items;
String prefix;
List<String> filteredList = items.stream().filter(e -> (!e.startsWith(prefix))).collect(Collectors.toList());
5. 日期相关API(Java 8 Date/Time API Changes)
新的日期和时间API /类(JSR-310),也称为ThreeTen,仅更改了Java应用程序中处理日期的方式。
日期
Date类过时了,已经被新的类LocalDate,LocalTime和LocalDateTime所取代。
- LocalDate类取代了data。没有时间或时区的表示。
- LocalTime类取代了time。没有日期或时区的表示。
- LocalDateTime类取代了date-time。没有时区的表示。
如果需要使用带时区信息的日期功能,可以用OffsetDate,OffsetTime和OffsetDateTime。
时区偏移可以以“ +05:30”或“Europe/Paris”格式表示。这可以通过使用另一个类ZoneId来完成。
LocalDate localDate = LocalDate.now();
LocalTime localTime = LocalTime.of(12, 20);
LocalDateTime localDateTime = LocalDateTime.now();
OffsetDateTime offsetDateTime = OffsetDateTime.now();
ZonedDateTime zonedDateTime = ZonedDateTime.now(ZoneId.of("Europe/Paris"));
时间戳和持续时间
某一个特定的时间点可以使用 Instant 类来表示,Instant 类也可以用来创建旧版本的java.util.Date 对象。
Instant instant = Instant.now();
Instant instant1 = instant.plus(Duration.ofMillis(5000));
Instant instant2 = instant.minus(Duration.ofMillis(5000));
Instant instant3 = instant.minusSeconds(10);
Duration类是Java新提出来的的概念。它表示两个时间戳之间的时差
Duration duration = Duration.ofMillis(5000);
duration = Duration.ofSeconds(60);
duration = Duration.ofMinutes(10);
Period可以处理较小的时间单位,例如毫秒,秒,分钟和小时
Period period = Period.ofDays(6);
period = Period.ofMonths(6);
period = Period.between(LocalDate.now(), LocalDate.now().plusDays(60));
本文来自转载,原文链接: