从集合到流

接上一小节初探Lambda表达式/Java多核编程【0】从外部迭代到内部迭代,本小节将着手使用“流”这一概念进行“迭代”操作。

首先何为“迭代”。其意为对某一对象进行某种操作,并把结果作为下一次操作的输入,如此往复进行,直至得到或逼近预期结果。

现在我们用代码来具体表示对某一集合进行迭代操作,我们希望定义一个Contact类来表示联系人,并将ContactList中所有String类型的联系人姓名全部包装进Contact类中:

List<Contact> contacts = new ArrayList<>();
contactList.forEach(new Consumer<String>() {
    @Override
    public void accept(String s) {
        Contact contact = new Contact();
        contact.setName(s);
        contacts.add(contact);
    }
});

接下来我们希望筛选出所有还能打通的联系人,将其放入一个有效联系人集合:

List<Contact> validContacts = new ArrayList<>();
contacts.forEach(new Consumer<Contact>() {
    @Override
    public void accept(Contact c) {
        if (c.call())
            validContacts.add(c);
    }
});
System.out.println(validContacts.size());

可以看出,第一次操作我们将String类型的数据转换为Contact,第二次则对每一个Contact调用call()方法,筛选出返回结果为true的联系人并将其收集进另一个集合,最后我们统计出还能打通的联系人数目。

在此过程中,操作行为完全封闭在各个集合内部,无需引入任何外部变量。

从处理开始、进行到结束,对象在操作间如同一个有序序列在移动,这就是流的特征,即“移动中的数据”。

真正的流与集合大相径庭,其只表示一种“可选的有序值序列”,而“无需为这些值提供任何存储”,这就是为何Stream在Java8-API中被定义为接口而非一种类。

public interface Stream<T> extends BaseStream<T, Stream<T>> {}

Stream<T>为对象的流,而DoubleStreamLongStream以及IntStream则为double、long以及int这三种基本类型的流。

现在我们再将第一次从String到Contact的映射用流的方式来重写:

Stream<Contact> contactStream = contactList.stream().map(s -> new Contact().setName(s));

stream()从源中取得管道,表示流的开始。

map()接收管道中的流并对齐进行某种变换,在本例中,我们将管道中的String映射成为Contact类,自此,String管道成为Contact管道。

我们可以将上一段代码拆分为:

Stream<String> stringStream = contactList.stream();
Stream<Contact> contactStream1 = stringStream.map(s -> new Contact().setName(s));

在基本搞清了流操作之后,我们现在一气呵成,直接使用流得到最终结果:

long validContactCounter =
    contactList.stream()
        .map(s -> new Contact().setName(s))
        .filter(c -> c.call())
        .count();

可以看出,我们对流能够进行丰富的操作,过滤、计数、查找等等,在此不表。

小结

使用流的方式处理数据能够精简代码,同时突出了所要进行的操作,当然乍看以来有些难懂。

既然牺牲了些许可读性,但是作为交换条件,我们在这种顺序执行的流操作中,获得了两倍于相应的循环版本的性能。

同样,并行执行流操作对于大型数据集将产生非凡的效果。

本小结相关代码:

(Contact.java)

```java

import java.util.Random;

public class Contact {

private String name;

private long number;

private Random random;

public Contact() {
    random = new Random();
}

public String getName() {
    return name;
}

public Contact setName(String name) {
    this.name = name;
    return this;
}

public long getNumber() {
    return number;
}

public Contact setNumber(long number) {
    this.number = number;
    return this;
}

public boolean call() {
    return random.nextBoolean();
}

}

(运行用)java

List

//--- Stream is coming ---//

Stream

//--- Break this code ---//

Stream

//--- All in one ---//

long validContactCounter =

contactList.stream()

.map(s -> new Contact().setName(s))

.filter(c -> c.call())

.count();

System.out.println(validContactCounter);

以及运行结果:java

3

3

```

初探Lambda表达式/Java多核编程【1】从集合到流的更多相关文章

  1. 初探Lambda表达式/Java多核编程【2】并行与组合行为

    今天又翻了一下书的目录,第一章在这之后就结束了.也就是说,这本书所涉及到的新的知识已经全部点到了. 书的其余部分就是对这几个概念做一些基础知识的补充以及更深层次的实践. 最后两个小节的内容较少,所以合 ...

  2. 初探Lambda表达式/Java多核编程【3】Lambda语法与作用域

    接上一篇:初探Lambda表达式/Java多核编程[2]并行与组合行为 本节是第二章开篇,前一章已经浅显地将所有新概念点到,书中剩下的部分将对这些概念做一个基础知识的补充与深入探讨实践. 本章将介绍L ...

  3. 初探Lambda表达式/Java多核编程【4】Lambda变量捕获

    这周开学,上了两天感觉课好多,学校现在还停水,宿舍网络也还没通,简直爆炸,感觉能静下心看书的时间越来越少了...寒假还有些看过书之后的存货,现在写一点发出来.加上假期两个月左右都过去了书才看了1/7都 ...

  4. 初探Lambda表达式/Java多核编程【0】从外部迭代到内部迭代

    开篇 放假前从学校图书馆中借来一本书,Oracle官方的<精通Lambda表达式:Java多核编程>. 假期已过大半才想起来还没翻上几页,在此先推荐给大家. 此书内容及其简洁干练,如果你对 ...

  5. Lambda&Java多核编程-5-函数式接口与function包

    从前面的总结中我们知道Lambda的使用场景是实现一个函数式接口,那么本篇就将阐述一下何为函数式接口以及Java的function包中提供的几种函数原型. 函数式接口 早期也叫作SAM(Single ...

  6. Lambda&Java多核编程-6-方法与构造器引用

    在Lambda&Java多核编程-2-并行与组合行为一文中,我们对Stream<Contact>里的每一位联系人调用call()方法,并根据能否打通的返回结果过滤掉已经失效的项. ...

  7. Lambda表达式和函数式编程

    Lambda表达式和函数式编程 https://www.cnblogs.com/bigbigbigo/p/8422579.html https://www.runoob.com/java/java8- ...

  8. Lambda&Java多核编程-7-类型检查

    本篇主要介绍Lambda的类型检查机制以及周边的一些知识. 类型检查 在前面的实践中,我们发现表达式的类型能够被上下文所推断.即使同一个表达式,在不同的语境下也能够被推断成不同类型. 这几天在码一个安 ...

  9. Lambda 表达式(C# 编程指南) 微软microsoft官方说明

    Visual Studio 2013 其他版本 Lambda 表达式是一种可用于创建委托或表达式目录树类型的匿名函数. 通过使用 lambda 表达式,可以写入可作为参数传递或作为函数调用值返回的本地 ...

随机推荐

  1. CentOSx64 安装 Gearmand 和 Gearman php扩展

    1.首先下载所需软件   wget https://github.com/downloads/libevent/libevent/libevent-2.0.20-stable.tar.gz    wg ...

  2. mac 访问mysql客户端

    /usr/local/mysql/bin/mysql -u root -p //mac mysql 管理工具推荐 sequek pro

  3. Lucene学习注意要点

    相关书籍: <Lucene实战>第二版: <搜索引擎基础教程>: <Lucene搜索引擎开发进阶实战>:(我现在看得书) 学习注意要点: 不要盲目从代码入手,而要先 ...

  4. css01

      1.  常用属性:①color:文本颜色:②background-color:背景颜色:③font-size:文字大小: 2.  样式声明:①内部样式:style=“样式规则:”,写于作用标签内, ...

  5. 关于div宽度和高度的100%设定

    设置DIV大小的有两个属性width和height,以前在学习DIV每次给DIV设置100%宽度或高度时都很迷惑,不明白这个100%的宽度(高度)到底有多宽有多高?这个100%是从哪里得到的从哪里继承 ...

  6. css 弹出框

    最近想弄一个类似登陆框的那种弹出框,其实网上已经有很多例子,而且也有相应的插件,例如:jquery-ui的,可直接使用,而我就简单的弄了个简易版的登陆框,真的很简易. 其实原理就是设置两个div,一个 ...

  7. libusb开发者指南(转)

    源:libusb开发者指南 译者: gashero 作者: Johannes Erdfelt 日期: 2010-04-17 地址: http://libusb.sourceforge.net/doc/ ...

  8. 今天学习了下,如何破解wifi

    破解了隔壁的wif,得罪了,哥哥要蹭网一段时间. 主要思路:安装linux环境.这里我选择了虚拟机加cdlinux .运行里面的min抓得握手包,然后淘宝花10元跑下包,因为自己的电脑不行,跑费时间太 ...

  9. Yii实现Password Repeat Validate Rule

    在使用Yii时遇到这样的需求:在一个注册的页面输入两次密码,并验证两次输入是否一致.可是password的repeat的字段在数据库 并不存在.问题来了,如何创建一个password_repeat的属 ...

  10. diskpart修改盘符

    开机运行一批处理.内容如下:diskpart /s c:\disk.txt c:\disk.txt内容如下:select disk 1            #1选择第二个硬盘 0选择第二个硬盘sel ...