1. 人人编程首页
  2. 编程语言
  3. Java

Java 8新特性介绍

本篇文章列出了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表达式的规则:

  1. Lambda表达式可以具有零个,一个或多个参数。
  2. 参数的类型可以显式声明,也可以从上下文中推断出来。
  3. 多个参数用强制括号括起来,并用逗号分隔。空括号用于表示一组空参数。
  4. 当存在单个参数时,如果推断出其类型,则不强制使用括号。例如a->返回a * a。
  5. Lambda表达式的主体可以包含零个,一个或多个语句。
  6. 如果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所取代。

  1. LocalDate类取代了data。没有时间或时区的表示。
  2. LocalTime类取代了time。没有日期或时区的表示。
  3. 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));

本文来自转载,原文链接:

发表评论

您的电子邮箱地址不会被公开。 必填项已用*标注