Java 函数式编程
由 JS 转 Java,写惯了 React,习惯了函数式,因此转 Java 时也是先学函数式。
语法糖「Syntactic Sugar」
起初,Java 的函数式看起来是匿名类的一个语法糖。
Stream.of(1, 2, 3).filter(new Predicate<Integer>() {
@Override
public boolean test(Integer integer) {
return integer % 2 == 0;
}
}).collect(Collectors.toList());
Stream.of(1, 2, 3).filter(x -> x % 2 == 0).collect(Collectors.toList());
后来,看到也能传方法引用时,我陷入了怀疑。
Stream.of(1, 2, 3).forEach(System.out::println);
StringBuilder sb = new StringBuilder();
Stream.of(1, 2, 3).forEach(sb::append);
System.out.println(sb.toString());
Stream.of(4, 2, 3).sorted(Integer::compareTo);
不过再想想,还是可以理解成语法糖:
Stream.of(1, 2, 3).forEach(System.out::println);
Stream.of(1, 2, 3).forEach(new Consumer<Integer>() {
@Override
public void accept(Integer integer) {
System.out.println(integer);
}
});
闭包「Closure」
Java 闭包也是假的,和匿名类的限制一样,要求闭包访问的外部作用域变量是 final
的。实现应该都是:基础类型值传递,引用类型引用传递。
以下 Java 代码编译会报错:
int index = 0;
Stream.of("a", "b", "c")
.forEach(x -> System.out.println((index++) + ":" + x));
Error: Variable used in lambda expression should be final or effectively final
JS 闭包毫无问题:
let index = 0;
["a", "b", "c"].forEach(x => console.log(index++ + ":" + x));
尽管如此,但函数式习惯还是可以带到 Java 了。
第一个要带过来的是:少量的数据结构搭配大量的操作。
在 OOP 的世界里,开发者被鼓励针对具体的问题建立专门的数据结构,并以方法的形式,将专门的操作关联在数据结构上。函数式编程语言选择了另一种重用思路。它们用很少的一组关键数据结构( 如 list、 set、 map)来搭配专为这些数据结构深度优化过的操作。我们在这些关键数据结构和操作组成的一套运转机构上面,按需要“ 插入” 另外的数据结构和高阶函数来调整机器,以适应具体的问题。
Neal Ford. 函数式编程思维
在 Java 中,关键数据结构就是指 Stream
。
Stream 操作三板斧:map
、filter
、reduce
Java 和 JS 有些不一样。
不同点一:index
的取法
JS map
、filter
、reduce
都能拿到 index
:
["a", "b", "c"]
.map((s, index) => `${index}: ${s}`)
.forEach(s => console.log(s));
Java 想要 index
的信息,如果只是单线程,可以考虑利用引用类型保存迭代的索引:
AtomicInteger i = new AtomicInteger(0);
Stream.of("a", "b", "c")
.map(x -> i.getAndIncrement() + ":" + x)
.forEach(System.out::println);
如果想支持多线程,可以先通过 zip
让流中的数据带上 index
:
public static <T, R> List<Pair<T, R>> zip(List<T> s1, List<R> s2) {
return IntStream.range(0, Math.min(s1.size(), s2.size()))
.mapToObj(index -> new Pair<T, R>(s1.get(index), s2.get(index)))
.collect(Collectors.toList());
}
public static void main(String[] args) {
List<String> list = Arrays.asList("a", "b", "c");
zip(IntStream.range(0, list.size()).boxed().collect(Collectors.toList()), list)
.parallelStream()
.map(p -> p.getKey() + ":" + p.getValue())
.forEach(System.out::println);
}
不同点二:reduce
的用法
Java 分为了 reduction
和 mutable reduction
,在 JS 里是不区分的。
A mutable reduction operation accumulates input elements into a mutable result container, such as a Collection or StringBuilder, as it processes the elements in the stream.
JS
reduction
:[1, 2, 3].reduce((acc, cur) => acc + cur, 0);
Java
reduction
:Stream.of(1, 2, 3).reduce(0, Integer::sum);
JS
mutable reduction
:[1, 2, 2].reduce((acc, cur) => {
acc.add(cur);
return acc;
}, new Set());
Java
mutable reduction
:Stream.of(1, 2, 2).collect(
() -> new HashSet<>(),
(set, el) -> set.add(el),
(s1, s2) -> s1.addAll(s2)
);
也可以简写为:
Stream.of(1, 2, 2).collect(
HashSet::new,
HashSet::add,
HashSet::addAll
);
参数比 JS 多了个
HashSet::addAll
是为了并行处理。
后续继续总结高阶函数、柯里化、函子在 Java 中的应用。
Java 函数式编程的更多相关文章
- paip.提升效率---filter map reduce 的java 函数式编程实现
#paip.提升效率---filter map reduce 的java 函数式编程实现 ======================================================= ...
- 《深入理解Java函数式编程》系列文章
Introduction 本系列文将帮助你理解Java函数式编程的用法.原理. 本文受启发于JavaOne 2016关于Lambda表达式的相关主题演讲Lambdas and Functional P ...
- Java函数式编程原理以及应用
一. 函数式编程 Java8所有的新特性基本基于函数式编程的思想,函数式编程的带来,给Java注入了新鲜的活力. 下面来近距离观察一下函数式编程的几个特点: 函数可以作为变量.参数.返回值和数据类型. ...
- Java函数式编程:一、函数式接口,lambda表达式和方法引用
Java函数式编程 什么是函数式编程 通过整合现有代码来产生新的功能,而不是从零开始编写所有内容,由此我们会得到更加可靠的代码,并获得更高的效率 我们可以这样理解:面向对象编程抽象数据,函数式编程抽象 ...
- Java函数式编程:二、高阶函数,闭包,函数组合以及柯里化
承接上文:Java函数式编程:一.函数式接口,lambda表达式和方法引用 这次来聊聊函数式编程中其他的几个比较重要的概念和技术,从而使得我们能更深刻的掌握Java中的函数式编程. 本篇博客主要聊聊以 ...
- Java函数式编程:三、流与函数式编程
本文是Java函数式编程的最后一篇,承接上文: Java函数式编程:一.函数式接口,lambda表达式和方法引用 Java函数式编程:二.高阶函数,闭包,函数组合以及柯里化 前面都是概念和铺垫,主要讲 ...
- Java函数式编程和lambda表达式
为什么要使用函数式编程 函数式编程更多时候是一种编程的思维方式,是种方法论.函数式与命令式编程的区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做.说白了,函数式编程是基于 ...
- 精练代码:一次Java函数式编程的重构之旅
摘要:通过一次并发处理数据集的Java代码重构之旅,展示函数式编程如何使得代码更加精练. 难度:中级 基础知识 在开始之前,了解"高阶函数"和"泛型"这两个概念 ...
- [译]通往 Java 函数式编程的捷径
原文地址:An easier path to functional programming in Java 原文作者:Venkat Subramaniam 译文出自:掘金翻译计划 以声明式的思想在你的 ...
- Java 函数式编程(Lambda表达式)与Stream API
1 函数式编程 函数式编程(Functional Programming)是编程范式的一种.最常见的编程范式是命令式编程(Impera Programming),比如面向过程.面向对象编程都属于命令式 ...
随机推荐
- centos7 离线升级/在线升级操作系统内核
目录 一.前言 二.系统环境 三.系统内核下载网址 四.centos7离线升级系统内核 1.先查看系统环境 2.离线升级系统内核 五.在线升级系统内核 一.前言 CentOS(Community EN ...
- React技巧之表单提交获取input值
正文从这开始~ 总览 在React中,通过表单提交获得input的值: 在state变量中存储输入控件的值. 在form表单上设置onSubmit属性. 在handleSubmit函数中访问输入控件的 ...
- Nacos配置失败(java.lang.IllegalStateException: failed to req API:/nacos/v1/ns/instance after all server)
解决: nacos服务器过载,可以删掉nacos文件夹下的data文件夹,重新启动.
- 一切皆为字节和字节输出流_OutputStream类&FileOutputStream类介绍
一切皆为字节 一切文件数据(文本.图片.视频等)在存储时,都是以二进制数字的形式保存,都一个一个的字节,那么传输时一样如此.所以,字节流可以传输任意文件数据.在操作流的时候,我们要时刻明确,无论使用什 ...
- HMS Core音频编辑服务支持7种音频特效,助力一站式音频处理
多媒体时代,音频作为内容传播中的重要形式,因其不受空间限制.认知负担小.声音元素多样化等特点,广泛应用于短视频制作.儿童在线教育.有声阅读.游戏等领域产品,在各种形式的音频呈现过程中,合理添加音效能够 ...
- 记一次实战 Shiro反序列化内网上线
Shiro反序列化内网上线 说明:此贴仅分享用于各安全人员进行安全学习提供思路,或有合法授权的安全测试,请勿参考用于其他用途,如有,后果自负.感谢各位大佬的关注 目标:152.xxx.xxx.xxx目 ...
- 弹性布局( display: flex;)
参考: https://www.cnblogs.com/hellocd/p/10443237.html
- 6.4.2 用BFS求最短路
前面的篇幅占了太多,再次新开一章,讲述BFS求最短路的问题 注意此时DFS就没有BFS好用了,因为DFS更适合求全部解,而BFS适合求最优解 这边再次提醒拓扑变换的思想在图形辨认中的重要作用,需要找寻 ...
- 基于CentOS的IB网卡驱动安装
一.准备步骤 1.查看centos版本 cat /etc/issue或者cat /etc/redhat-release 2.查看linux版本 cat /proc/version或uname -a或 ...
- raspberrypi系统在加入k8s作为node节点时遇到的问题
新买的树莓派4b到货后就迫不及待的烧录上raspberrypi系统,将新派加入我的k8s集群,期间遇到了点小挫折,好歹也一个一个解决了: 一.kubelet版本不对导致无法加入k8s集群 在执行kub ...