GitHub Page: http://blog.cloudli.top/posts/Java-函数式编程-流操作/

外部迭代到内部迭代

在使用集合类时,通用的方式是在使用 for 循环集合上进行迭代,然后处理每一个元素。比如计算集合中来自上海的用户:

int count = 0;
for (User user : allUsers) {
if (user.isFrom("Shanghai")) {
count++;
}
}

以上代码为外部迭代,这样的方式要写很多样板代码。

另一种方式是内部迭代:

long count = allUsers.stream()
.filter(user -> user.isFrom("Shanghai"))
.count();

整个过程变成了函数调用,且被分解为两个简单的操作:

  • 找出所有来自上海的用户。
  • 计算他们的数量。

每种操作都对应 Stream 接口的一个方法。为了找出来自上海的用户,需要对 Stream 对象进行过滤:filter() 方法,该方法接受一个 lambda 表达式,最后由 count() 方法计算给定的 Stream 中包含多少对象。

只调用 filter() 方法并不会做任何操作:

allUsers.stream()
.filter(user -> {
out.println(user.name);
return user.isFrom("Shanghai")
});

以上代码不会产生任何输出,只有最后调用了 count() 方法,才能看到输出。filter 方法返回的是一个 Stream 对象,这个过程和建造者模式相似。

建造者模式通过一系列的操作设置属性,只有最后调用了 build() 方法对象才会被创建。在这里 filter() 方法只是传递了一个 lambda 表达式,只有在 count() 方法被调用时才会去执行它。

常用的流操作

filter

filter() 方法在上一部分已经出现过,它使用给定的 lambda 表达式检查流中的元素,并过滤出符合条件的元素。

List<User> usersFromShanghai = allUsers.stream()
// 过滤出来自上海的用户
.filter(user -> user.isFrom("Shanghai"))
.collect(Collectors.toList());

filter() 方法的参数是 Predicate 接口:

collect

该方法接受一个 Collector 类型的函数式接口。可以结合 Collectors 提供的方法来完成操作。

将所有来自上海的用户存入一个列表中:

List<User> usersFromShanghai = allUsers.stream()
.filter(user -> user.isFrom("Shanghai"))
.collect(Collectors.toList());

统计每个地区用户的数量,这个操作与 SQL 中的 group by 类似。:

Map<String, Long> userMap = allUsers.stream()
// 以 User 的 location 进行分组,对每个分组计数
.collect(Collectors.groupingBy(User::getLocation, Collectors.counting()));

collect() 方法还可以完成很多操作,以上只是其中一种。

map

map() 可以将一个流中的值转换为一个新的流。

将一个字符串列表全部转为大写:

List<String> list = Stream.of("a", "b", "c")
// 将每个字符串转换成大写
.map(str -> str.toUpperCase())
// 转换为列表
.collect(Collectors.toList());

这里传给 map() 的 lambda 表达式接受一个 String 类型的参数,返回一个新的 String。参数和返回值可以不属于同一种类型。

map() 方法的参数是 Function 函数式接口:

flatMap

flatMap() 将一个流中每一个值都转换为 Stream 对象。

可以使用该方法将多个列表合并:

List<Integer> together = Stream.of(Arrays.asList(1, 2, 3), Arrays.asList(4, 5, 6))
// 将每个列表都转换成 Stream
.flatMap(numbers -> numbers.stream())
// 转换成列表
.collect(Collectors.toList());

flatMap() 方法的参数也是 Function 接口,唯一的区别在于返回值限定为 Stream 类型。

reduce

reduce() 方法可以实现从一组值中生成一个值。

对一个列表进行求和:

int sum = Stream.of(1, 2, 3, 4)
.reduce(0, (acc, e) -> acc + e);

对一个列表求阶乘:

int sum = together.stream()
.reduce(1, (acc, e) -> acc * e);

Java 函数式编程--流操作的更多相关文章

  1. Java函数式编程:三、流与函数式编程

    本文是Java函数式编程的最后一篇,承接上文: Java函数式编程:一.函数式接口,lambda表达式和方法引用 Java函数式编程:二.高阶函数,闭包,函数组合以及柯里化 前面都是概念和铺垫,主要讲 ...

  2. Java函数式编程原理以及应用

    一. 函数式编程 Java8所有的新特性基本基于函数式编程的思想,函数式编程的带来,给Java注入了新鲜的活力. 下面来近距离观察一下函数式编程的几个特点: 函数可以作为变量.参数.返回值和数据类型. ...

  3. Java函数式编程:二、高阶函数,闭包,函数组合以及柯里化

    承接上文:Java函数式编程:一.函数式接口,lambda表达式和方法引用 这次来聊聊函数式编程中其他的几个比较重要的概念和技术,从而使得我们能更深刻的掌握Java中的函数式编程. 本篇博客主要聊聊以 ...

  4. 《深入理解Java函数式编程》系列文章

    Introduction 本系列文将帮助你理解Java函数式编程的用法.原理. 本文受启发于JavaOne 2016关于Lambda表达式的相关主题演讲Lambdas and Functional P ...

  5. Java函数式编程:一、函数式接口,lambda表达式和方法引用

    Java函数式编程 什么是函数式编程 通过整合现有代码来产生新的功能,而不是从零开始编写所有内容,由此我们会得到更加可靠的代码,并获得更高的效率 我们可以这样理解:面向对象编程抽象数据,函数式编程抽象 ...

  6. paip.提升效率---filter map reduce 的java 函数式编程实现

    #paip.提升效率---filter map reduce 的java 函数式编程实现 ======================================================= ...

  7. Java 函数式编程(Lambda表达式)与Stream API

    1 函数式编程 函数式编程(Functional Programming)是编程范式的一种.最常见的编程范式是命令式编程(Impera Programming),比如面向过程.面向对象编程都属于命令式 ...

  8. Java 函数式编程

    由 JS 转 Java,写惯了 React,习惯了函数式,因此转 Java 时也是先学函数式. 语法糖「Syntactic Sugar」 起初,Java 的函数式看起来是匿名类的一个语法糖. Stre ...

  9. Java函数式编程和lambda表达式

    为什么要使用函数式编程 函数式编程更多时候是一种编程的思维方式,是种方法论.函数式与命令式编程的区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做.说白了,函数式编程是基于 ...

随机推荐

  1. 单词chalchiguite硬玉chalchiguite英语

    翡翠在我国明确地称为硬玉(chalchiguite) ,可能始于宋代.19世纪后半叶,法国矿物学家德穆尔将中国的“玉”分为软玉和硬玉(见地质情报所:<翡翠冲的<国外地质科技>,198 ...

  2. Violet音乐社区 - 总结报告

    目录 一.项目概述 1.1 项目背景 1.2 编写目的 1.3 项目文档 1.4 项目源码 1.5 项目成果 二.个人工作总结 1.1 工作概览 1.1.1 作为项目组组长 1.1.2 作为项目组成员 ...

  3. To B产品,业务方全程蒙蔽怎么搞?

            这是发生在很久前的事,那会我还是产品实习生.         今天和业务部门进行需求审核,对的是公司内部SAAS系统的采购模块.怎么说呢?就是觉得不专业吧         辛辛苦苦把原 ...

  4. JDBC及C3P0常用类

    JDBC(Java Database Connectivity)JAVA数据库连接,它是一套用于执行SQL语句的Java API.JDBC可以通过不同驱动与不同数据库连接,相当于JAVA和数据库之间的 ...

  5. Mongodb基础 学习小结

    MongoDB是一个基于分布式文件存储的数据库.由C++语言编写.旨在为WEB应用提供可扩展的高性能数据存储解决方案.MongoDB是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能 ...

  6. javascript加载XML字符串或文件

    1. 加载XML文件 方法1:ajax方式.代码如下: var xhr = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObje ...

  7. python27期day04:列表、元组、range、作业题。

    1.for循环套for循环: for i in "abc": for x in "egf: print(x) 结果是:e g f e g f e g f  2.99乘法表 ...

  8. http协议里定义的四种常见数据的post方法

    原文 https://blog.csdn.net/charlene0824/article/details/51199292 关于http协议里定义的四种常见数据的post方法,分别是: applic ...

  9. golang和python互相调用

    http://blog.yuanzhaoyi.cn/2018/06/27/golang_python.html python3-ctypes: https://docs.python.org/3.5/ ...

  10. zzL4自动驾驶中感知系统遇到的挑战及解决方案

    本次分享的大纲: Perception Introduction Sensor Setup & Sensor Fusion Perception Onboard System Percepti ...