Java8新特性
Posted 仲翎逸仙
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java8新特性相关的知识,希望对你有一定的参考价值。
原文出处:http://blog.decaywood.me/about/#zh
整个教程将详细的介绍Java8所有的新特性,并以简短的代码展示出来。你将学会如何使用默认接口方法/lambda表达式/方法引用以及@Repeatable注解。读完本文后,你将对流(stream)/函数式接口(functional interface)/map拓展以及新的日期API等最新的API变动有一个相对了解。整个教程大部分都是以代码形式体现,没有过多的文字说明,让我们开始吧:)
接口默认方法
Java8可以让我们通过default关键字在接口中添加非抽象的方法,这是一个新的概念,虚拟拓展方法(补充:这样可以提供声明行为的默认实现)
话不多说,上代码:
interface Formula {
double calculate(int a);
default double sqrt(int a) {
return Math.sqrt(a);
}
}
除了calculate抽象方法,Formula接口同时定义了默认方法sqrt。这样一来,接口实现类只需要实现抽象方法calculate即可,而默认方法sqrt则可以在实现类外部进行调用。代码如下:
Formula formula = new Formula() {
@Override
public double calculate(int a) {
return sqrt(a * 100);
}
};
formula.calculate(100); // 100.0
formula.sqrt(16); // 4.0
上述代码中,Formula接口由匿名对象实现。可以看出,代码非常冗长,仅仅简单的实现了sqrt(a * 100)计算就用了6行代码。我们将在下一章节看到如何在Java8中用更简洁的方式实现单个抽象方法的接口。
以下为补充:
Java 8带来的另一个有趣的特性是接口可以声明(并且可以提供实现)静态方法。例如:
public interface DefaulableFactory {
// Interfaces now allow static methods
static double calculate(int a) {
return sqrt(a * 100);
}
}
在JVM中,默认方法的实现是非常高效的,并且通过字节码指令为方法调用提供了支持。默认方法允许继续使用现有的Java接口,而同时能够保障正常的编译过程。这方面好的例子是大量的方法被添加到java.util.Collection接口中去:stream(),parallelStream(),forEach(),removeIf(),……
尽管默认方法非常强大,但是在使用默认方法时我们需要小心注意一个地方:在声明一个默认方法前,请仔细思考是不是真的有必要使用默认方法,因为默认方法会带给程序歧义,并且在复杂的继承体系中容易产生编译错误。更多详情请参考官方文档
Lambda表达式
让我们以一个简单老版本的字符串排序为例子开始这一章节:
List<String> names = Arrays.asList("peter", "anna", "mike", "xenia");
Collections.sort(names, new Comparator<String>() {
@Override
public int compare(String a, String b) {
return b.compareTo(a);
}
});
静态排序方法Collections.sort接收一个List和一个比较器来对List里面的元素进行排序,你可能经常遇到这种需要传入一个匿名比较器给排序方法的情况。
除了创建一个匿名对象的方式来实现Comparator接口之外,Java8提供了一个更加简洁的语法,lambda表达式:
Collections.sort(names, (String a, String b) -> {
return b.compareTo(a);
});
很明显,代码明显简洁了许多而且表达更加清晰了,但它还可以简化:
Collections.sort(names, (String a, String b) -> b.compareTo(a));
对于只有一行的方法,可以采取去掉外围方法体和返回值的方式进一步简化:
names.sort((a, b) -> b.compareTo(a));
List现在新增了sort方法,并且编译器可以自动分析出形参类型,所以你可以直接省略形参类型。接下来,我们会更加深入探索lambda表达式的各种神奇的用法。
函数式接口
lambda表达式是如何适应Java的类型系统的呢?每种给定类型的lambda表达式由一个接口来进行描述。这就是所谓的函数式接口,函数式接口必须只包含一个抽象方法定义。每个对应的lambda表达式唯一匹配这一个抽象方法。由于默认方法非抽象,你可以在函数式接口中随意添加默认方法。
只要接口只含有唯一的抽象方法,我们就可以把它用作lambda表达式。为了确定你的方法能满足函数式接口的要求,建议加上@FunctionalInterface注解。编译器会通过这个注解来进行编译检查,只要你在被注解标记的接口中定义了第二个抽象方法,编译器在编译时就会抛出一个异常。
例子:
@FunctionalInterface
interface Converter<F, T> {
T convert(F from);
}
Converter<String, Integer> converter = (from) -> Integer.valueOf(from);
Integer converted = converter.convert("123");
System.out.println(converted); // 123
请一定要记住@FunctionalInterface注解不是必须的,如果省略,代码依然是有效的。
方法引用和构造器引用
如果使用静态方法引用,上文中的例子能够得到更进一步的简化。
Converter<String, Integer> converter = Integer::valueOf;
Integer converted = converter.convert("123");
System.out.println(converted); // 123
Java8允许你通过使用::关键字来传递方法和构造器的引用。上面的例子展示了如何引用一个静态方法。不过我们也可以引用实例方法:
class Something {
String startsWith(String s) {
return String.valueOf(s.charAt(0));
}
}
Something something = new Something();
Converter<String, String> converter = something::startsWith;
String converted = converter.convert("Java");
System.out.println(converted); // "J"
让我们看看::关键字是如何引用构造器的。首先我们定义一个有不同构造器的Person类:
class Person {
String firstName;
String lastName;
Person() {}
Person(String firstName, String lastName) {
this.firstName = firstName;
this.lastName = lastName;
}
}
接下来我们定义一个Person工厂接口用于创建Person对象:
interface PersonFactory<P extends Person> {
P create(String firstName, String lastName);
}
除了像以前那样新建一个实现类来实现接口以外,我们还可以通过构造器引用把接口和构造器两者关联起来:
PersonFactory<Person> personFactory = Person::new;
Person person = personFactory.create("Peter", "Parker");
我们通过Person::new创建了一个指向Person构造器的引用。Java编译器会自动对代码进行分析最后选择一个对应的构造器来匹配PersonFactory接口中定义的方法签名即PersonFactory.create。
Lambda 作用域
从lambda表达式访问外部作用域的变量和匿名对象有点类似。你可以从本地外部作用域访问带有final修饰符的变量,也可以访问实例变量和静态变量。
访问本地变量
我们可以从lambda表达式的外部作用域读取本地final变量:
final int num = 1;
Converter<Integer, String> stringConverter =
(from) -> String.valueOf(from + num);
stringConverter.convert(2); // 3
但不同于匿名对象,num变量如果不声明为final,代码依然是有效的:
int num = 1;
Converter<Integer, String> stringConverter =
(from) -> String.valueOf(from + num);
stringConverter.convert(2); // 3
然而,虽然不需要定义num为final,但是隐式地必须为不变量才能通过编译检查,下面的代码就不能通过编译,因为num值被修改了:
int num = 1;
Converter<Integer, String> stringConverter =
(from) -> String.valueOf(from + num);
num = 3;
同时,从lambda表达式赋值给num也是禁止的。(补充:对于这种行为很好理解,本地变量存储在栈中,lambda表达式本质上还是对象,存储在堆中,必然不能对栈数据进行写操作。)
访问域变量和静态变量
相比本地变量,对于实例变量和静态变量,lambda表达式既有读权限也有写权限,这种行为在匿名对象中非常常见。
class Lambda4 {
static int outerStaticNum;
int outerNum;
void testScopes() {
Converter<Integer, String> stringConverter1 = (from) -> {
outerNum = 23;
return String.valueOf(from);
};
Converter<Integer, String> stringConverter2 = (from) -> {
outerStaticNum = 72;
return String.valueOf(from);
};
}
}
访问默认接口方法
你应该还记得第一章提到的Formula例子吧?Formula接口定义了一个默认方法sqrt,它能被Formula接口的实现类或者匿名对象调用。但在lambda表达式中则不能调用默认方法。下面的代码不能通过编译:
Formula formula = (a) -> sqrt( a * 100);
内置函数式接口
JDK1.8的API囊括了许多内置函数式接口。其中一些在老版本中就被广泛使用(补充:还记得上文说过的吗?只要接口中定义了唯一的抽象方法就符合函数式接口的规范,可以被当做函数式接口。),例如Comparator或者Runnable。这些老版本就存在的接口在新版本中经过@FunctionalInterface注解标记进行拓展以提供lambda支持。
除了这些接口,Java8 API也提供了全新的函数式接口来减轻你的工作负担。其中一些接口借鉴了Google Guava库中被广泛使用的功能。即使你对Guava了如指掌,你也应该认真地了解一下Java8中提供的新接口拓展了哪些有用的方法。
Predicates
Predicate接口是一个返回boolean值的接口,它接收一个参数。这个接口包含了各种默认方法,可以对Predicates进行各种组合来完成复杂的逻辑运算(and, or, negate)。
Predicate<String> predicate = (s) -> s.length() > 0;
predicate.test("foo"); // true
predicate.negate().test("foo"); // false
Predicate<Boolean> nonNull = Objects::nonNull;
Predicate<Boolean> isNull = Objects::isNull;
Predicate<String> isEmpty = String::isEmpty;
Predicate<String> isNotEmpty = isEmpty.negate();
Functions
Function接口接收一个参数,返回一个结果。其包含的默认方法能用于将多个Functions结构链接起来(compose,andthen)
Function<String, Integer> toInteger = Integer::valueOf;
Function<String, String> backToString = toInteger.andThen(String::valueOf);
backToString.apply("123"); // "123"
Suppliers
Supplier接口根据给定的泛型返回一个结果。不像Functions接口,Suppliers不需要传入参数。
Supplier<Person> personSupplier = Person::new;
personSupplier.get(); // new Person
Consumers
Consumer接口代表一类只接收一个参数的操作。
Consumer<Person> greeter = (p) -> System.out.println("Hello, " + p.firstName);
greeter.accept(new Person("Luke", "Skywalker"));
Comparators
Comparator接口在老版本JDK中就广泛使用。Java8新引入了许多默认方法到接口中。
Comparator<Person> comparator = (p1, p2) -> p1.firstName.compareTo(p2.firstName);
Person p1 = new Person("John", "Doe");
Person p2 = new Person("Alice", "Wonderland");
comparator.compare(p1, p2); // > 0
comparator.reversed().compare(p1, p2); // < 0
Optionals
Optional虽然不属于函数式接口,但它是防止空指针错误(NullPointerException)的利器。Optional对于接下来的内容来说是一个非常重要的概念,所以,让我们先大概地了解一下Optional是如何工作的。
Optional是一个存储值的简单容器,只有null和non-null两种概念。想象一个方法,它有可能返回一个非空的结果,但也有可能什么都不返回(补充:例如查找某个用户,但是并没有这个用户的相关信息,null会造成空指针错误,所以返回null按道理是不合适的)。在Java8中,你这时就可以返回Optional来替代null了。
Optional<String> optional = Optional.of("bam");
optional.isPresent(); // true
optional.get(); // "bam"
optional.orElse("fallback"); // "bam"
optional.ifPresent((s) -> System.out.println(s.charAt(0))); // "b"
Streams
java.util.Stream代表一系列可以进行一种或多种操作的元素,Stream也叫做流。流操作既可以是intermediate(补充:即还有下一步操作)也可以是terminal(终止)的。terminal操作返回一个特定类型的结果。intermediate操作则返回流对象自身以进行链式调用。流由source(数据源)创建,例如集合(java.util.Collection)中的List/Set(map不支持流操作)。流操作既可以顺序地执行也可以并行执行(补充:这里指并行流)。
有需要的可以看一看Stream.js,一个跟Java8 Stream API风格相似的javascript API。
让我们看看顺序流是如何工作的。首先我们创建一个String列表作为Stream的示例源:
List<String> stringCollection = new ArrayList<>();
stringCollection.add("ddd2");
stringCollection.add("aaa2");
stringCollection.add("bbb1");
stringCollection.add("aaa1");
stringCollection.add("bbb3");
stringCollection.add("ccc");
stringCollection.add("bbb2");
stringCollection.add("ddd1");
Collections在Java8中进行了拓展,所以你可以通过调用Collection.stream()或者Collection.parallelStream()很方便地生成流。接下来的章节中将会介绍最常用的一些流操作。
Filter
Filter接收一个Predicate类型的参数来对流中的所有元素进行过滤操作。Filter操作属于intermediate操作,可以让我们在输出结果后进一步调用其他流操作(forEach)。ForEach操作接收一个Consumer类型的参数来处理经过Filter操作后的流的元素。ForEach属于terminal操作。它没有返回值,所以我们不能继续流操作。
stringCollection
.stream()
.filter((s) -> s.startsWith("a"))
.forEach(System.out::println);
// "aaa2", "aaa1"
Sorted
Sorted操作属于intermediate操作,它返回一个排好序的流的视图。如果不传入自定义的Comparator,元素将以默认排序规则排序。
stringCollection
.stream()
.sorted()
.filter((s) -> s.startsWith("a"))
.forEach(System.out::println);
// "aaa1", "aaa2"
注意,sorted操作只是创建一个排好序的流的视图而不是操作内部隐藏的集合元素顺序。stringCollection的顺序被没有被篡改:
System.out.println(stringCollection);
// ddd2, aaa2, bbb1, aaa1, bbb3, ccc, bbb2, ddd1
Map
intermediate操作map传入Function对象将流中的每个元素转换为另一个对象。下面的例子演示了将每个String对象转换为大写的String对象。你也可以使用map操作来将流中的元素转换为其他类型的对象。返回的流的泛型类型依赖于你传给map的Function对象的泛型类型(补充:你不需要显式指定泛型类型,Java8的编译器相当智能,可以自动推导出参数类型)。
stringCollection
.stream()
.map(String::toUpperCase)
.sorted((a, b) -> b.compareTo(a))
.forEach(System.out::println);
// "DDD2", "DDD1", "CCC", "BBB3", "BBB2", "AAA2", "AAA1"
Match
不同的match操作可以用于判断特定的predicate是否匹配流中的元素。所有这种类型的操作都属于terminal操作,以返回一个boolean值结束。
boolean anyStartsWithA =
stringCollection
.stream()
.anyMatch((s) -> s.startsWith("a"));
System.out.println(anyStartsWithA); // true
boolean allStartsWithA =
stringCollection
.stream()
.allMatch((s) -> s.startsWith("a"));
System.out.println(allStartsWithA); // false
boolean noneStartsWithZ =
stringCollection
.stream()
.noneMatch((s) -> s.startsWith("z"));
System.out.println(noneStartsWithZ); // true
Count
Count属于terminal操作,返回流中的元素个数,类型为long。
long startsWithB =
stringCollection
.stream()
.filter((s) -> s.startsWith(&q以上是关于Java8新特性的主要内容,如果未能解决你的问题,请参考以下文章