Java8 Stream API 简介

先看一段代码

List<String> names = Arrays.asList("Jack", "Jill", "Nate", "Kara", "Kim", "Jullie", "Paul", "Peter");
 
List<String> subList = new ArrayList<>();
for(String name : names) {
  if(name.length() == 4)
    subList.add(name);
}
 
StringBuilder namesOfLength4 = new StringBuilder();
for(int i = 0; i < subList.size() - 1; i++) {
  namesOfLength4.append(subList.get(i));
  namesOfLength4.append(", ");
}
        
if(subList.size() > 1)
  namesOfLength4.append(subList.get(subList.size() - 1));
 
System.out.println(namesOfLength4);

Stream API的历史

  • 在java8引入
  • 受益于lambda表达式

lambda表达式

接口常被用于传递代码,如sort接收一个Comparator接口用于给文件数组按名称排序

Arrays.sort(files, new Comparator<File>() {

    @Override
    public int compare(File f1, File f2) {
        return f1.getName().compareTo(f2.getName());
    }
});

不过上述代码需要new一个匿名类,而实际上sort()方法只应该知道如何对两个文件做比较即可,也就是说只要一个函数体,但是java却非要一个类!
Java 8提供只需要传入函数即可进行排序的方案,这就是lambda表达式:

Arrays.sort(files, (f1, f2) -> f1.getName().compareTo(f2.getName())); 

上面的代码就自然多了,只需给出一个函数就能完成排序,(f1, f2)表示传入的参数,类型由java根据前面的files自行推断出来。如果无参数,则可以写成()

函数式接口

接口替换为lambda表达式不是无条件的,要求该接口是所谓的函数式接口

函数式接口就是一个具有一个方法的普通接口

可以用@FunctionalInterface来注解某个接口,强行要求某个接口为函数式接口,如果该接口含有2个方法,则编译会失败;默认方法静态方法并不影响函数式接口的契约,可以任意使用:

@FunctionalInterface
public interface Comparator<T> {
  int compare(T o1, T o2);
  default Comparator<T> reversed() {
      return Collections.reverseOrder(this);
  }
  public static <T extends Comparable<? super T>> Comparator<T> 
  reverseOrder() {
      return Collections.reverseOrder();
  }
  ......
}

一句话,lambda表达式的目的是让大家编程不用再记忆类名函数名,也不需要写参数类型,自然得传入需要的函数体即可。
比如这段sort()这段代码:

Arrays.sort(files, (f1, f2) -> f1.getName().compareTo(f2.getName())); 

根本不需要记忆sort需要传入Comparator接口,也不需要记忆它的方法名称compare(),也不需要写参数类型File,方便吧。

Stream API

针对常见的集合数据处理,Java 8引入了一套新的类库,位于包java.util.stream下,称之为Stream API。Java 8给Collection接口增加默认方法,可以返回一个Stream。

default Stream<E> stream() {
    return StreamSupport.stream(spliterator(), false);
}

在介绍Stream API之前,先看2个函数式接口,定义在包java.util.function下:

image.png

可以把它们想象提供和Comparator类似的功能,就是提供一个函数体。
Predicate叫谓词函数,测试输入是否满足条件,返回truefalse
Function叫函数变换,将输入值1->1映射为另一个值。
我们完全不用在意它们的接口名和方法名,只要知道它们的用途即可。

为便于举例,我们先定义一个简单的学生类Student,有name和score两个属性,如下所示,我们省略了getter/setter方法。

static class Student {
    String name;
    double score;
    
    public Student(String name, double score) {
        this.name = name;
        this.score = score;
    }
}

有一个学生列表:

List<Student> students = Arrays.asList(new Student[] {
        new Student("zhangsan", 89d),
        new Student("lisi", 89d),
        new Student("wangwu", 98d) });

map 变换

image.png

返回学生的姓名列表:

List<String> nameList = students.stream()
        .map(s->s.getName())
        .collect(Collectors.toList());

map()接收一个Function接口,将学生变换为姓名;在collect做实际的处理,将学生对象挨个变换为姓名,并构成列表输出。

filter变换

image.png

过滤得分在90以上的学生:

List<Student> above90List = students.stream()
        .filter(t->t.getScore()>90)
        .collect(Collectors.toList());

filter()接收一个Predicate接口,过滤出90分以上的学生,在collect做实际的处理,构成列表输出。

复合变换

返回90分以上的学生姓名:

  1. 过滤:得到90分以上的学生列表
  2. 转换:将学生列表转换为姓名列表
List<String> above90Names = students.stream()
        .filter(t->t.getScore()>90)
        .map(s->s.getName())
        .collect(Collectors.toList());

效率问题探讨

对复合变换,如果用旧方法,会在一次遍历中做过滤和变换操作:

List<String> above90NamesList = new ArrayList<>();
for (Student t : students) {
    if (t.getScore() > 90) {
        above90NamesList.add(t.getName());
    }
}

而用Stream的方式看起来似乎先filter遍历了一次,map又遍历了一次,这样速度岂不会变慢?其实不会,java的Stream用了一种巧妙(tricky)的技术,实现了惰性求值,直到最后一步collect的时候才会做遍历操作。要向做到这一点,应该采用某种方式记录用户每一步的操作,当用户调用结束操作时将之前记录的操作叠加到一起在一次迭代中全部执行掉,具体实现可以看这篇文章:深入理解Java Stream流水线

stream操作分类

前面讲到filter,map,collect都是stream的方法,但是它们是有所不同的,stream的操作分为2类中间操作(intermediate operations)结束操作(terminal operations),归纳如下:

操作类型 接口方法
中间操作 concat() distinct() filter() flatMap() limit() map() peek() skip() sorted() parallel() sequential() unordered()
结束操作 allMatch() anyMatch() collect() count() findAny() findFirst()forEach() forEachOrdered() max() min() noneMatch() reduce() toArray()
  • 中间操作总是会惰式执行,调用中间操作只会生成一个标记了该操作的新stream,仅此而已。
  • 结束操作会触发实际计算,计算发生时会把所有中间操作积攒的操作以流水线的方式执行,这样可以减少迭代次数。计算完成之后stream就会失效。

更多的例子

不仅是List有stream操作,Set也有steam操作,下面是个例子:

Map<Integer, String> HOSTING = new HashMap<>();
        HOSTING.put(1, "linode.com");
        HOSTING.put(2, "heroku.com");
        HOSTING.put(3, "digitalocean.com");
        HOSTING.put(4, "aws.amazon.com");

//Map -> Set -> Stream -> Filter -> List
List<String> strings = HOSTING.entrySet().stream()
            .filter(map -> "aws.amazon.com".equals(map.getValue()))
            .map(map -> map.getValue())
            .collect(Collectors.toList());
System.out.println(strings);

回到开头的例子

List<String> names = Arrays.asList("Jack", "Jill", "Nate", "Kara", "Kim", "Jullie", "Paul", "Peter");
         
// 给定一个名称集合,仅选择长度为 4 的名称,然后通过逗号将它们连接起来。
System.out.println(
  names.stream()
    .filter(name -> name.length() == 4)
    .collect(Collectors.joining(", ")));

问题:给定名为 numbers 的列表,此代码将计算大于 3 且小于 8 的偶数并将该数字乘以 2,然后输出结果。

int result = 0;
for(int e : numbers) {
  if(e > 3 && e % 2 == 0 && e < 8) {
    result += e * 2;
  }
}
System.out.println(result);
System.out.println(
 numbers.stream()
   .filter(e -> e  > 3)
   .filter(e -> e % 2 == 0)
   .filter(e -> e < 8)
   .mapToInt(e -> e * 2)
   .sum());

参考文献

Java Stream API入门篇
深入理解Java Stream流水线
计算机程序的思维逻辑 (91) - Lambda表达式
计算机程序的思维逻辑 (92) - 函数式数据处理 (上)
Java 8 Stream Tutorial
提倡使用有帮助的编码

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容

  • Streams 原文链接: Streams 原文作者: shekhargulati 译者: leege100 状态...
    忽来阅读 5,522评论 3 32
  • Java8 in action 没有共享的可变数据,将方法和函数即代码传递给其他方法的能力就是我们平常所说的函数式...
    铁牛很铁阅读 1,227评论 1 2
  • 第一章 为什么要关心Java 8 使用Stream库来选择最佳低级执行机制可以避免使用Synchronized(同...
    谢随安阅读 1,490评论 0 4
  • lambda表达式(又被成为“闭包”或“匿名方法”)方法引用和构造方法引用扩展的目标类型和类型推导接口中的默认方法...
    183207efd207阅读 1,477评论 0 5
  • 金色的梅印0106阅读 169评论 0 0