1、新时代的Java语言
新时代的Java是以JDK8的发布为分水岭,从JDK8开始,Java开发进入一个新的阶段,Java8 经历9个里程碑版本,于2014年3月18日正式发布,距今已有很长一段时间了,很多企业已经逐步迁移Java8,Java8有非常重大的变化,新增了很多新的语法与特性;
一些流行框架都采用了Java8,比如Mybatis3.5.x开始全面采用JDK8,Spring5.x开始全面采用JDK8,SpringBoot2.x开始全面采用JDK8,Spring Cloud F版开始也要求JDK8;
2、Java8的一些新变化(统计)
195新的文件增加到了JDK8(目前JDK总共4240文件)
93个新类, 89新接口,13新的枚举
2699新方法, 56新的构造函数,49新属性
46接口被注解为@FunctionalInterface(函数式接口)
213接口默认方法
68静态接口方法
3、Java8有哪些新特性?
1、接口默认方法
2、函数式接口
3、Lambda表达式
4、Stream流式API
5、Optional
6、新日期处理
7、......
4、Java8新特性
新特性1:接口默认方法和静态方法
接口默认方法:
接口不仅可以声明未实现的抽象方法,也可以声明已经实现了的方法,而不需要实现类去实现其方法。
默认方法就像一个普通Java方法,只是方法的修饰符用default关键字。
接口静态方法
接口中可以定义静态方法,它与普通的静态方法没有区别。
静态方法就像一个普通Java静态方法,但方法的关键词只能是public或者不写。
为什么要新增该特性?
如果JDK在java8时,在List接口中新增了一个抽象方法,而我们项目写了一个类实现了List接口,由于项目是之前采用jdk7开发的,那么当我们项目采用jdk8运行时,就会报错,因为你没有实现List里面新的抽象方法。
所以Java设计人员引入了接口默认方法,其目的是为了解决接口的修改与已有的实现不兼容的问题,接口默认方法可以作为库、框架向前兼容的一种手段。
新特性2:函数式接口
只包含一个方法的接口(指抽象方法)叫函数式接口;
Java标准库中的java.lang.Runnable, java.util.concurrent.Callable就是典型的函数式接口。
在Java 8中通过@FunctionalInterface注解,将一个接口标注为函数式接口,该接口只能包含一个方法。
@FunctionalInterface注解不是必须的,只要接口只包含一个方法,虚拟机会自动判断该接口为函数式接口。
一般建议在接口上使用@FunctionalInterface注解进行声明,以免他人错误地往接口中添加新方法,如果在你的接口中定义了第二个抽象方法的话,编译器会报错。
但是标记了@FunctionalInterface注解的接口可以有一个或多个默认接口方法和静态方法,比如Java类库中的: java.util.Comparator 。
函数式接口里允许定义java.lang.Object里的public方法(非final方法)
Java 8中类库中许多接口都添加了@FunctionalInterface注解,比如:
java.lang.Runnable,
java.util.Comparator,
java.util.concurrent.Callable
java.util.function包下的所有接口
函数式接口是为Java 8中的lambda而设计的,lambda表达式的方法体其实就是函数接口的实现。
新特性3:Lambda表达式
Lambda 表达式是一个匿名函数,它是一个函数式接口的具体实现方式,使用Lambda语法可以用来代替Java8之前的匿名内部类,从而简化代码。
任意只包含一个抽象方法的接口(函数式接口),我们都可以用lambda表达式去编写。
Lambda表达式语法结构
包含三个部分
1、一个括号内用逗号分隔的形式参数,参数是函数式接口里面方法的参数;
2、一个箭头符号:->
3、一个方法体,可以是表达式和代码块,方法体是函数式接口里面方法的实现,如果Lambda的功能语句块比较复杂,我们可以用大括号包起来,跟普通的Java方法一样。
总体结构如下:
(parameters) -> expression 或者 (parameters) -> { statements; }
1、括号里的形式参数类型可以省略,编译器会根据上下文来推测参数的类型,你也可以显式地指定参数类型,如果没有形式参数,则括号里空着。
2、方法体,如果有多行功能语句用大括号括起来,如果只有一行功能语句则可以省略大括号。
3、如果Lambda方法体只有一行return语句,则如果省略了大括号,那么return可以省略;
4、如果参数列表只有一个参数,则参数列表的小括号可以省略;
新特性4:内置功能接口
Java8的内置功能接口在java.util.function包下,主要有四大核心的内置功能型接口:
谓词(Predicate)
函数(Function)
生产者(Supplier)
消费者(Consumer)
1、谓词:(Predicate)
谓词是单参数的返回布尔值的函数式接口,输入一个参数,返回true或者false。
2、函数:(Function)
Function接受一个参数,并产生一个结果。
3、生产者:(Supplier)
Suppliers产生一个给定的泛型类型的结果。与Function不同的是Suppliers不接受输入参数。
4、消费者:(Consumer)
Consumers代表在一个单一的输入参数上执行操作。 Consumer的操作可能会更改输入参数的内部状态。
新特性5:方法引用
方法引用是用来直接访问类或者实例的已经存在的方法或者构造方法。
方法引用是lambda表达式的一种简写形式。
如果lambda表达式只是调用一个特定的已经存在的方法,则可以使用方法引用。
如果你觉得lambda的方法体会很长,影响代码可读性,则可以用方法引用来解决。
语法:
使用 :: 操作符将方法名与对象或类的名字分隔开。
新特性6:Stream API
Stream是Java 8中引入的全新API,位于java.util.stream包下,它与 java.io 包下的 InputStream 和 OutputStream 等输入输出流是完全不同的概念。
Java 8 中的 Stream 是对集合(Collection)功能的增强,它专注于对集合对象进行各种非常便利高效的操作。它的底层实现使用了并行化操作, 利用Java7 中引入的 Fork/Join 框架来拆分任务,加速处理过程,充分利用了现在多核处理器的优势。
Stream分为中间操作和末端操作。
Filter
Filter 接受一个谓词来过滤出流中所有的元素。
Sorted
Sorted 是一个返回流的自然排序的中间运算。除非你传递一个定制的Comparator ,元素将被以自然顺序进行排序。
Map
Map是一个中间操作,可以将集合中每个元素通过给定的函数转变为其它对象。
Match
match是一个末端操作,可以被用来检验某stream是否与一个特定的谓词相匹配,返回一个布尔值结果。
Count
计数是一个末端运算,以long类型返回在stream中的元素的数目。Parallel Streams
我们知道,streams可以是顺序或者并行的。
在顺序streams上的操作是在一个单线程中完成的,而并行streams上的操作是在多个线程上并发完成的。
使用并行stream可以充值利用多核处理器的优势,提升速度和性能,而所有你所需要做的仅仅是将stream() 改为 parallelStream()即可。