当函数成为一等公民时,设计模式的变化

GOF提出的设计模式,其本质思想是封装变化。故而,创建型模式封装的是对象创建的变化,结构型模式封装的是对象之间的协作与组合结构,行为型模式则封装了对象行为的变化。所谓“行为”,不正是函数所能要表达的吗?

函数的抽象能力

从函数的抽象角度看,任何行为都可以理解为是一个对类型进行转换的函数,这是FP思想对OO设计模式的最大冲击。例如Strategy模式与Command模式,前者封装了算法策略的变化,后者则封装了命令请求的变化。无论算法策略,还是命令请求,都可以表现为一个函数。

譬如说将各种四则运算看做是一种算法策略,为了应对具体计算的变化,在Java中我们应该定义四则运算的策略接口:

public interface Strategy {  
    int compute(int a, int b);  
} 
public class Context  {  
    private final Strategy strategy;  
    public Context(Strategy strategy) { 
        this.strategy = strategy; 
    }  
    public void use(int a, int b) { 
        strategy.compute(a, b); 
    }  
}

接收两个整数,然后经过计算返回一个整数。显然,四则运算的调用者其实关注的不是Strategy这个接口,而是compute这个行为。跟进一步,调用者其实关注的是将两个整数转换为一个整数的行为,他并不关心接口是什么,函数名有是什么,而是关注f(a, b) = c这个函数。于是,在Scala中,策略模式的实现就变为:

class Context(f: (Int, Int) => Int) {  
  def use(a: Int, b: Int): Int =  f(a, b)  
}

当然,你可以可以为这个函数定义一个类型,使其更加表意:

type Stategy = (Int, Int) => Int

当然,如果面对的是一组策略行为的封装,且这些策略行为的变化是一致的,使用一个接口将这些行为封装起来,在重用和表意角度讲,似乎又比单纯使用函数更佳。Scala提供OO与FP两种范式,算是一种骑墙的取巧,程序员需要依势而为。Scala给你提供了丰富而精彩的食材,如果你没有将菜做得色香味俱全,不能怪食材不好,还是自己太烂了。

Scala还提供了一种类似block的机制,称之为by name call。它接受的是一个语句块,而非函数类型。所以要注意这种形式与无参函数的区别。此外,by name call同时还具有延迟调用的能力。例如,当我们定义一个invoke函数接受一个无传入参数的函数时:

def invoke(f: () => Unit) = f()

如果你向invoke传入println("scala"),scala会报告错误:

这是因为println("scala")返回的是Unit类型,而不是() => Unit函数类型。使用by name call就没有这个问题:

f: => Unit是一个语句块,所以不能像函数那样调用。我们可以使用这种方式来快捷实现Command模式。

由于Java 8已经支持Lambda表达式,虽然它仍然不支持高阶函数,但是作为Java程序员,仍然有必要培养函数抽象的能力与习惯。在Java 8中使用Lambda,不仅让语法变得简洁,还可以让调用者可以脱离对具体某个接口的依赖,而仅仅依赖函数的抽象特征。

函数的组合能力

FP的编程思想中,除了高阶函数(包括Curry等)具有的抽象能力之外,还有一个好处是提供组合子能力。落实到Scala的语法上,就是偏函数(Partial Function)的andThencomposeorElse

Pavel Fatin在文章《Design Patterns in Scala》用OO设计模式中的Chain of Responsibility(职责链)模式来对比组合子,其实还是比较牵强的。或者说,FP思想中的组合子远远比职责链模式更强大。在Elixir语言中,甚至还提供了管道操作符>|来实现这种函数的组合。而我在博客《Scala中的Partial Function》中已经非常详解地介绍了Scala的偏函数,大家可以移步阅读。

如果真要对比,那么结合Scala的语法来看,则orElse可以非常方便地模拟职责链模式,而andThen则近似于管道-过滤器模式。其实我在OO语言中,很少运用GOF标志的职责链模式,也就是当寻找到具体职责的承担者时,履行职责后即可退出的方式;而是对这种模式进行调整,让其在履行职责后继续执行next的职责,又近乎于管道-过滤器了。

所以说,设计模式的运用妙乎于心,讲究应势而变。在融入FP思想后,要从本质思想去面对这些模式,不拘泥于OO还是FP,似乎才是未来编程的取舍之道。

原文发布于微信公众号 - 逸言(YiYan_OneWord)

原文发表时间:2017-03-03

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java呓语

工厂方法模式(延迟到子类来选择实现)

1、工厂方法模式理念介绍 2、它与简单方法模式的区别 3、推荐使用工厂方法的场景 4、在Android 源码中的应用

8840
来自专栏ACM算法日常

最高的牛Tallest Cow(前缀和)- POJ 3263

FJ's N (1 ≤ N ≤ 10,000) cows conveniently indexed 1..N are standing in a line. E...

12610
来自专栏青玉伏案

PHP精选数组函数

编程怎么能少的了数组呢,以下是学习PHP时常用的数组处理函数。在编程中要遵循一个原则就是DRY(Don`t Repeat Yourself)原则,PHP中有大...

24980
来自专栏大史住在大前端

javascript基础修炼(2)——What's this(上)

this是javascript关键字之一,是javascript能够实现面向对象编程的核心概念。用得好能让代码优雅高端,风骚飘逸,用不好也绝对是坑人坑己利器。我...

9910
来自专栏web前端教室

js数据结构与算法--散列

不扯淡了,还是来学技术吧。 散列,是一种常用的数据存储技术,优势在于可以快速的插入或取出,使用它的数据结构,叫散列表。 它的优势哈,插入、删除、取用数据都很快,...

240100
来自专栏程序员互动联盟

【编程基础】main函数,你知道多少?

近期学习时对这个问题产生了迷惑,看到了这篇文章,感觉挺好。 在C/C++的学习过程中,一个很常见的问题就是void main和int main有什么区别呢?本文...

37580
来自专栏Petrichor的专栏

什么是:语法糖、语法盐、语法糖精

57650
来自专栏向治洪

Scala入门笔记

Scala入门 Scala简介 ps:在最新的薪资调查中,Scala程序员的工资是平均最高的Scala工资。 Scala是一门多范式的编程语言,一种类似ja...

22570
来自专栏C语言及其他语言

[每日一题]平移运动

估计大家今天忙开学迎新什么的都忙不过来了吧,今天介绍的这题呢,跟之前的题很像,也是数组的题 题目描述 有n个整数,使前面各数顺序向后移m个位置,最后m个数变成...

28450
来自专栏Java帮帮-微信公众号-技术文章全总结

第八天 自定义类型方法集合混合使用【悟空教程】

21380

扫码关注云+社区

领取腾讯云代金券