编者注:您也可以在此处检查Part-1。
嗨,朋友们,这是简明系列的Java 8语言功能的第2部分。
在这里,我们将讨论Java 8的以下功能:
- 接口中的静态方法
- 流
1.接口中的静态方法
什么是静态方法?
静态方法是属于类而不是对象或实例的方法。 换句话说,这意味着该类的所有实例共享在静态方法中定义的相同行为。 因此,您无需创建Class的实例或对象即可调用静态方法。
因此,如果静态方法不需要调用对象,则将它们放在类中就没有多大意义,因为类是属于该类的所有对象的蓝图。 那么如果我们可以将所有静态方法放在接口而不是类中呢? 好消息是,Java 8允许这样做。 使用Java 8,我们可以将以前用来放置类的实用程序方法放入接口中。
接口中静态方法的语法
语法与其他任何静态方法一样。
static <Return Type> <Method Name>();
接口中静态方法的主要目的
允许在接口中添加静态方法的主要原因是在接口中而不是在类中具有实用程序方法。 在Java 8之前,如果我们要定义一些实用程序方法,那么我们将创建一个类并在该类中定义实用程序方法,因为我们知道创建仅包含实用程序方法的类的对象是没有意义的,因此我们正在构造构造函数该类的名称为Private,并将该类标记为final,以使其无法继承,因为我们知道静态方法不会被继承,因此最好将其显式标记,以便仅查看该类就可以知道类不能被继承。
接口中的静态方法示例
在Java 8之前
final class CustomerUtil {private CustomerUtil(){}public static Date convertDateFormat(){}public static String appendData() {}
}
class Test {public static void main(String[] args){CustomerUtil.convertDateFormat(); }
}
Java 8之后
interface CustomerUtil {public static Date convertDateFormat(){}public static String appendData() {}
}
class Test {
public static void main(String[] args){CustomerUtil.convertDateFormat(); }
}
2.流
Java 8中的Stream是什么
流是源中支持聚合操作的一系列元素。 因此,每个流将有一个源和一个目的地。 数据通过Stream从源流向目标。
下面的Oracle文档摘录很好地描述了Collection和Stream之间的区别。
- 没有存储 。 流不是存储元素的数据结构。 相反,它通过一系列计算操作从数据结构,数组,生成器功能或I / O通道等源中传递元素。
- 本质上是功能性的 。 对流的操作会产生结果,但不会修改其源。 例如,一种过滤
Stream
获得从集合产生一个新的Stream
,而不过滤元件,而不是删除从源收集元件。 - 懒惰 。 许多流操作(例如过滤,映射或重复删除)可以延迟实施,从而暴露出进行优化的机会。 例如,“使用三个连续的元音查找第一个
String
”不需要检查所有输入字符串。 流操作分为中间(Stream
)操作和终端(产生值或副作用)操作。 中间操作总是很懒。 - 可能是无限的 。 尽管集合的大小是有限的,但流不是必需的。 诸如
limit(n)
或findFirst()
类的短路操作可以允许对无限流的计算在有限时间内完成。 - 易耗品 。 在流的生存期内,流的元素只能访问一次。 像
Iterator
一样,必须生成新的流以重新访问源中的相同元素
Stream的语法
流可以通过多种方式创建。 我们将看到最常用的一种方法,并在另一篇文章中讨论其他方法。
从上图可以看出,在Collection接口中添加了新的默认方法stream()
,并且List,Set和Queue接口扩展了Collection接口,因此所有这些接口都具有stream方法,该方法可以用于从这些集合中创建Stream作为源。
使用列表作为源创建字符串流的语法
List<String> list = new ArrayList<>();
list.add("AA");Stream<String> stream = list.stream();
创建以Set为源的字符串流的语法
Set<String> set = new HashSet<>();
set.add("AA");Stream<String> stream = set.stream();
流的主要目的
在Java中引入Stream API的主要目的是减少冗长的代码来对诸如Collections,Array之类的一组数据执行操作,并利用多核处理器(使用并行流)而无需程序员编写一行多线程代码,从而提高了性能。
流操作示例
假设您有一个元素列表(字符串),并且想要一个不同的元素列表。 接下来,我们将看到您将如何在Java 8之前和Java 8上做到这一点。
在Java 8之前
List<String> list = new ArrayList<>();
list.add("AA");
list.add("BB");
list.add("BB");
list.add("BB");
System.out.println("Size of list before applying distinct logic:"+list.size());
List<String> distinctList1 = new ArrayList<String>();
for(String str : list){if(!distinctList1.contains(str)){distinctList1.add(str);}
}
System.out.println("Distinct List Size:"+ distinctList1.size());
Java 8之后
List<String> sourceList = new ArrayList<>();
sourceList.add("AA");
sourceList.add("BB");
sourceList.add("BB");
sourceList.add("BB");
System.out.println("Size of list before applying Stream Operations:"+sourceList.size());
List<String> distinctList = sourceList.stream().distinct().collect(Collectors.toList());
System.out.println("Distinct List Size:"+ distinctList.size());
在上面的代码中, sourceList
是Stream distinct()
的元素源, Stream distinct()
和collect()
是流操作。 distinctList
是destinition列表。
从上面的示例可以看出,使用Java 8,可对Collection进行操作的代码更像是查询,而不是很多样板代码。 就像在SQL中一样,我们不需要编写逻辑来从elements.Sql
集合中查找最大值,而Sql为此提供了max()
函数。 同样,Java 8流提供了许多聚合函数,我们也可以将它们链接在一起以在一行中完成多项操作。 这种方法的优点是该代码不太冗长,并且还利用了多核处理器,因此也提高了性能。
参考文献:
https://docs.oracle.com/javase/8/docs/api/java/util/stream/package-summary.html
http://www.oracle.com/technetwork/articles/java/ma14-java-se-8-streams-2177646.html
翻译自: https://www.javacodegeeks.com/2018/07/java-8-language-features-part-2.html