GC、Lambda、 Java8的流(Stream)概念其实都来自函数式编程。
他因何有如此魔力呢?
在函数式编程:
这种函数叫高阶函数(High-order function),就如高中数学的复合函数 f(g(x))。
高阶函数作用在于,可用它去做行为的组合:
find方法就是个高阶函数:接收一个函数作为参数,这样一些处理逻辑就能外置出去。
这段代码的使用者,就能按需组合。
高阶函数让代码编写方式出现质变:
这就是典型的函数式编程风格:
一个好模型的设计就是逐层叠加。所以函数式编程的组合性,就是好的设计方式。
但把模型拆解成多个可组合的构造块就很考验开发者分离关注点。这是智力上的超越,而大多数开发者都只会无脑crud而已。
把模型拆成小的构造块,构造块足够小,就会发现一些通用构造块。
函数式编程探索是从LISP语言开始,LISP源自“List Processing”,指明了这个语言的核心概念:List列表,最为常用的数据结构。
LISP认为大部分操作最后都可归为列表转换,即数据经过一系列的列表转换会得到一个结果。要理解这一系列转换,就要先理解每个基础的转换:map、filter和reduce等,MapReduce也就源自函数式编程里列表转换的模式。
若能正确理解,就能抛弃for循环。
比如,我有一组数[1、2、3、4]:
把一组数据通过一个函数映射为另一组数据。
经过map操作,这里用作映射的函数是乘以2,即这组数都乘2,就得到一组新数[2、4、6、8]。
把一组数据按照某个条件进行过滤,只有满足条件的数据才会留下。
过滤函数:大于2,即只有大于2的数才会留下,得到的结果就是[3、4]。
把一组数据按照某个规则,归约为一个数据。
做个reduce操作,其归约函数是一个加法操作,也就是这组数里面的每个元素相加,最终会得到一个结果,也就是 1+2+3+4=10。
现在想知道学生里男生总数,可给Student类新增性别字段:
传统写法:
按列表转换思维,首先,过程分解:
刚好对应map、filter、reduce:
分解后映射到代码上。Java 8也支持列表转换。为兼容原有API,提供了新接口Stream:,可将其理解成List的另一种表现形式。
于是使用Java8 Stream的写法:
基本和操作步骤对应,只是多了步将性别转换成1,便于后面计算。
map、filter和reduce只是最基础的三个操作,列表转换可提供操作要更多。只是大多数在这三基础封装。
比如,上面最后两步map、reduce,Java8 Stream接口提供了count:
同是处理一组数据,推荐函数式的列表转换,而非传统for循环:
结构化编程提供的控制结构也是一层封装。熟悉函数式编程后,这些代码理解起来同那些控制结构无本质区别,只是抽象级别更高,提供更好表达性。
代码的表达性,有一个描述了做什么的接口后,具体怎么做就可以在背后不断优化了。
比如,如果一个列表的数据特别多,可考虑并发处理,而这种优化对使用端透明。MapReduce 甚至将运算分散到不同的机器上执行,但背后逻辑都一样。
实际工作中如何将面向对象和函数式编程两种不同的编程范式组合运用。
一个好的函数式的接口,需分离关注点。虽然你不知道组合方式会有多少,但所有变化其实就是元素组合。
面向对象关键在于结构的组合,而函数式编程在于函数接口的组合。
将单纯结构化的功能代码,重构成了领域模型+应用层引用的方式。属于领域模型的功能内敛,应用层对这些功能的复杂性无感。同时在多个应用层间,该领域模型的功能都是可以复用的,不管是代码去重还是复用性都有不错的提高
版权说明 : 本文为转载文章, 版权归原作者所有 版权申明
原文链接 : https://blog.csdn.net/qq_33589510/article/details/120637222
内容来源于网络,如有侵权,请联系作者删除!