鱼C论坛

 找回密码
 立即注册
查看: 3774|回复: 0

[好文转载] Java 8:不要再用循环了

[复制链接]
发表于 2017-4-16 00:00:19 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
本帖最后由 零度非安全 于 2017-4-15 22:40 编辑
作者:ImportNew - 进林

正如我之前(http://www.deadcoderising.com/wh ... -lambdas-in-java-8/)所写的,Java 8 中的新功能特性改变了游

戏规则。对 Java 开发者来说这是一个全新的世界,并且是时候去适应它了。

在这篇文章里,我们将会去了解传统循环的一些替代方案。在Java 8的新功能特性中,最棒的特性就是允许我们去表达我们

想要完成什么而不是要怎样做。这正是循环的不足之处。要确保循环的灵活性是需要付出代价的。return、break 或者 con

tinue 都会显著地改变循环的实际表现。这迫使我们不仅要清楚我们要实现怎样的代码,还要了解循环是怎样工作的。在介

绍 Java 8 的流(Stream)时,我们学会了一些集合操作的实用技巧。现在我们要看看怎样把这些循环转换为更简洁,可读

性更高的代码。
0.png


开始编码!

好吧,讲的够多了,是时候展示一些例子了!

这次我们要以文章为例子。一篇文章拥有一个标题,一个作者和几个标签。
private class Article {
 
    private final String title;
    private final String author;
    private final List<String> tags;
 
    private Article(String title, String author, List<String> tags) {
        this.title = title;
        this.author = author;
        this.tags = tags;
    }
 
    public String getTitle() {
        return title;
    }
 
    public String getAuthor() {
        return author;
    }
 
    public List<String> getTags() {
        return tags;
    }
}
每个例子都会包含一个使用传统循环的方案和一个使用Java 8新特性的方案。

在第一个例子里,我们要在集合中查找包含“Java”标签的第一篇文章。

看一下使用for循环的解决方案。
public Article getFirstJavaArticle() {
    for (Article article : articles) {
        if (article.getTags().contains("Java")) {
            return article;
        }
    }
    return null;
}
现在我们使用Stream API的相关操作来解决这个问题。
public Optional<Article> getFirstJavaArticle() {
    return articles.stream()
    .filter(article -> article.getTags().contains("Java"))
    .findFirst();
}
是不是很酷?我们首先使用 filter 操作去找到所有包含 Java 标签的文章,然后使用 findFirst() 操作去获取第一次出现的文

章。因为 Stream 是“延迟计算”(lazy)的并且 filter 返回一个流对象,所以这个方法仅在找到第一个匹配元素时才会处

理元素。

现在,让我们获取所有匹配的元素而不是仅获取第一个。

首先使用for循环方案。
public List<Article> getAllJavaArticles() {
    List<Article> result = new ArrayList<>();
    for (Article article : articles) {
        if (article.getTags().contains("Java")) {
            result.add(article);
        }
    }
    return result;
}
使用Stream操作的方案。
public List<Article> getAllJavaArticles() {  
    return articles.stream()
    .filter(article -> article.getTags().contains("Java"))
    .collect(Collectors.toList());
}
在这个例子里我们使用collection操作在返回流上执行少量代码而不是手动声明一个集合并显式地添加匹配的文章到集合里

到目前为止还不错。是时候举一些突出Stream API强大的例子了。

根据作者来把所有的文章分组。

照旧,我们使用循环方案。
public Map<String, List<Article>> groupByAuthor() { 
    Map<String, List<Article>> result = new HashMap<>();
    for (Article article : articles) {
        if (result.containsKey(article.getAuthor())) {
            result.get(article.getAuthor()).add(article);
        } else {
            ArrayList<Article> articles = new ArrayList<>();
            articles.add(article);
            result.put(article.getAuthor(), articles);
        }
    }
    return result;
}
我们能否找到一个使用流操作的简洁方案来解决这个问题?
public Map<String, List<Article>> groupByAuthor() {  
    return articles.stream()
    .collect(Collectors.groupingBy(Article::getAuthor));
}
很好!使用 groupingBy 操作和 getAuthor 方法,我们得到了更简洁、可读性更高的代码。

现在,我们查找集合中所有不同的标签。

我们从使用循环的例子开始。
public Set<String> getDistinctTags() { 
    Set<String> result = new HashSet<>(); 
    for (Article article : articles) {
        result.addAll(article.getTags());
    }
    return result;
}
好,我们来看看如何使用Stream操作来解决这个问题。
public Set<String> getDistinctTags() {  
    return articles.stream()
    .flatMap(article -> article.getTags().stream())
    .collect(Collectors.toSet());
}
棒极了!flatmap 帮我把标签列表转为一个返回流,然后我们使用 collect 去创建一个集合作为返回值。

一切皆有可能

以上的就是如何使用可读性更高的代码代替循环的例子。务必仔细看看Stream API,因为这篇文章仅仅提到它的一些皮毛

而已。


更新

收到 solarfuse 和 dhruvgairola 的评论后,更新了 getDistinctTags() 例子,使用集合(Set)作为返回集合。




本帖被以下淘专辑推荐:

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-10 14:16

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表