JAVA8系列教程-简介
该Java 8教程列出了重要的Java 8功能,并提供了此发行版中引入的示例。所有功能均具有指向详细教程的链接,例如lambda表达式,Java流,功能接口和日期时间API更改。
Java SE 8于2014年初发布。在Java 8中,最受关注的功能是lambda表达式。它还具有许多其他重要功能,例如默认方法,流API和新的日期/时间API。让我们通过示例了解Java 8中的这些新功能。
1. Lambda表达式
Lambda表达式对于使用Scala等其他流行编程语言的我们很多人并不陌生。在Java编程语言中,Lambda表达式(或函数)只是一个匿名函数,即不带名称且不受标识符限制的函数。它们被精确地写在需要的地方,通常作为其他函数的参数。
lambda表达式的基本语法为:
either (parameters) -> expression or (parameters) -> { statements; } or () -> expression |
典型的lambda表达式示例如下所示:
(x, y) -> x + y //This function takes two parameters and return their sum. |
请注意,根据x和y的类型,方法可能会在多个地方使用。参数可以匹配int或Integer或简单地也可以匹配String。根据上下文,它将添加两个整数或连接两个字符串。
编写Lambda表达式的规则
- Lambda表达式可以具有零个,一个或多个参数。
- 参数的类型可以显式声明,也可以从上下文中推断出来。
- 多个参数用强制括号括起来,并用逗号分隔。空括号用于表示空参数集。
- 当有单个参数时,如果推断出其类型,则不强制使用括号。例如a->return 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()
方法的行为,那么它可以提供自己的自定义实现并覆盖该方法。
重做更多:Java 8系列教程-默认方法
4. Java 8 Streams
Java 8 Streams API引入了另一个重大变化,该Java 8 Streams API提供了一种以多种方式处理一组数据的机制,这些方式可以包括过滤,转换或任何其他可能对应用程序有用的方式。
Java 8中的Streams API支持不同类型的迭代,您可以在其中简单地定义要处理的项目集,对每个项目执行的操作,以及存储这些操作的输出。
流API的示例。在此示例中,items
是String
值的集合,您要删除以某些前缀文本开头的条目。
List<String> items; String prefix; List<String> filteredList = items.stream().filter(e -> (!e.startsWith(prefix))).collect(Collectors.toList()); |
此处items.stream()
表示我们希望items
使用Streams API处理集合中的数据。
阅读更多:Java 8系列教程-内部与外部迭代
5. Java 8 Date/Time API的变更
新的日期和时间API /类(JSR-310),也称为ThreeTen,仅更改了您在Java应用程序中处理日期的方式。
日期
Date
类甚至已经过时了。旨在取代Date类新的类LocalDate
,LocalTime
和LocalDateTime
。
- 本
LocalDate
类代表一个日期。没有时间或时区的表示。 - 该
LocalTime
级表示时间。没有日期或时区的表示。 - 本
LocalDateTime
类代表一个日期-时间。没有时区的表示。
如果您希望使用区域信息的日期功能,那么拉姆达为您提供额外的3类类似于一个,即上面OffsetDate
,OffsetTime
和OffsetDateTime
。时区偏移可以以“ +05:30”或“欧洲/巴黎”格式表示。这是通过使用另一个类即完成的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" )); |
时间戳和持续时间
为了随时表示特定的时间戳,需要使用class Instant
。本Instant
类表示时间纳秒的精度瞬间。即时操作包括与另一个进行比较Instant
以及增加或减少持续时间。
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 ); |
Duration
处理较小的时间单位,例如毫秒,秒,分钟和小时。它们更适合与应用程序代码进行交互。与人互动,你需要得到更大的持续时间它都带有Period
类。
Period period = Period.ofDays( 6 ); period = Period.ofMonths( 6 ); period = Period.between(LocalDate.now(), LocalDate.now().plusDays( 60 )); |
- 本文标签: 其他
- 本文链接: https://www.v8en.com/article/228
- 版权声明: 本文由SIMON原创发布,转载请遵循《署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0)》许可协议授权