四个程序员的一天

函数编程的魅力与对比面向对象编程

你,一个DotNet程序员,刚刚加入一个新项目组。除了你之外,其他的成员包括:Ceer,一直从事C项目的程序员,他刚刚转入C#不到一个月; Jally,整天抱着本Design Pattern(没错,就是GoF的那本)在啃的前Java程序员;以及Semon,你对他完全不了解,只是听PM介绍说他是搞Scheme的(传说中的第二古老的语言LISP的方言之一)。不过你也没在意,毕竟计算机这玩意,老东西是不吃香的。

周一,刚打开电脑,老板就跑到你们组的办公座面前:“好吧,伙计们,现在有个function需要你们来搞定。具体是这样的:用户输入2个数,并输入一个操作符。你根据输入的情况来得出相应的运算结果。

1 Example: Foo(+, 1, 2) = 3; Foo(*, 3, 6) = 18; Foo(/, 2, 4) = 0.5

Ceer最先作出反应:简单嘛,判断一下输入的操作符就好了。说着,他很快在白板上写出如下代码: 

01 public classCStyle_Calculator   
02 {   
03  static public double Foo(char op, double x, doubley)   
04  {   
05   switch(op)   
06    case ’+’: returnx + y; break;   
07    case ’-’: returnx - y; break;   
08    case ’*’: returnx * y; break;   
09    case ’/’: returnx / y; break;   
10    default: throw newException(”What the Hell you have input?");   
11  }   
12 }

Jally只看了一遍,就捂着鼻子连连摇头:好一股的代码臭味。还不如看我用OO的方法来解决:

01 public interfaceI操作符 //谁说代码不能写中文的?恩恩   
02 {   
03  double 运算(double x, doubley);   
04 }   
05 public classOO_Calculator   
06 {   
07  privateI操作符 m_op;   
08  publicOO_Calculator(I操作符 op)   
09  {   
10   this.m_op = op; //依赖注入【注2】   
11  }   
12      
13  public double Foo(double x, doubley)   
14  {   
15   returnthis.m_op.运算(x, y);   
16  }   
17 }   
18      
19 public class加法:I操作符   
20 {   
21  public double 运算(double x, doubley)   
22  {   
23   returnx + y;   
24  }   
25 }   
26      
27 public class减法:I操作符   
28 {   
29  public double 运算(double x, doubley)   
30  {   
31   returnx - y;   
32  }   
33 }   
34      
35 public class乘法:I操作符   
36 {   
37  public double 运算(double x, doubley)   
38  {   
39   returnx * y;   
40  }   
41 }   
42      
43 public class除法:I操作符   
44 {   
45  public double 运算(double x, doubley)   
46  {   
47   returnx / y;   
48  }   
49 }   
50      
51 public classTheMainClass   
52 {   
53  static public voidMain()   
54  {   
55   I操作符 我的加法 = new加法();   
56   OO_Calculator 我的加法器 = newOO_Calculator(我的加法);   
57   doublesum = 我的加法器.Foo(3, 4);   
58   System.Console.WriteLine(sum);   
59   //sum = 7   
60      
61   //其他3个我就不废话了   
62  }   
63 }

你看着Jally把白板写得密密麻麻之后,耸耸肩,暗叹,你们这些用java的废柴,就一个运算器还搞出Interface这些东西,烦不烦啊。 让你们见识见识DotNet的强大吧. 那个运算符我直接用delegate传进去不就好了么.

01 public delegate double TheOperator(double x, doubley);   
02      
03 public classOperators   
04 {   
05  static public double Add(double x, doubley)   
06  {   
07   returnx + y;   
08  }   
09      
10  static public double Sub(double x, doubley)   
11  {   
12   returnx - y;   
13  }   
14      
15  //乘,除法 我也懒得废话了   
16 }   
17      
18 public classDotNet_Calculator   
19 {   
20  public double Foo(TheOperator op, double x, doubley)   
21  {   
22   returnop(x, y);   
23  }   
24 }   
25      
26 public classTheMainClass   
27 {   
28  static public voidMain()   
29  {   
30   TheOperator myAdd = newTheOperator(Operators.Add);   
31   TheOperator mySub = newTheOperator(Operators.Sub);   
32      
33   DotNet_Calculator dc = newDotNet_Calculator();   
34   doublesum = dc.Foo(myAdd, 2, 4); //sum = 6   
35   System.Console.WriteLine(sum);   
36   doublesub = dc.Foo(mySub, 3, 7); //sub = -4   
37   System.Console.WriteLine(sub);   
38  }   
39 }

//dot net 下面还可以用CodeDom动态构造C#代码,然后在内存编译运行。

//如果觉得专门写个Operators很烦的话,可以试试C#2.0的匿名方法 

很好,当你写完代码之后,挑衅的看着Jally,Ceer却开始抱怨起来:”这不就是C里面的函数指针么,我也会...“

“然则DotNet下面的Delegate是类型安全滴...”你继续洋洋得意./

而Semon,看了看你们3位华丽的代码,啥也没说,只是在键盘上敲下了2行代码

1 (define (Foo op x y)   
2 (op x y))

然后就下班了...

【注: scheme的代码稍微解释下:(+ 1 2) = 3, (* 3 4) = 12.】

至于Semon的解法:

1 (define (Foo op x y)   
2 (op x y))

看明白了么,上面的代码只有一个作用:第一行是函数头,定义了一个叫Foo的函数。该函数接受3个参数op, x, y。

第二行定义了函数的行为:把第一个参数op当作运算符,计算后面2个参数。

所以:(Foo + 1 2) = 3. (Foo / 12 6) = 2.

好了好了,不编故事了。

我只是想简单的让大家在繁忙的工作之余,也瞅瞅Function Programming(函数编程)世界的美妙。函数编程,最大的特点是它是将函数作为语言里1st class的元素来对待的。一个函数可以接受另一个函数作为参数,也可以把一个函数作为结果来返回。这样的函数我们称为Higher-order function。

那么,Function Programming和我们传统的面向对象有啥区别捏? 恩,这个嘛,扯得远可以扯到图灵机和冯·诺以曼这2种体系的差异...@_@不过那个太学术性,俺就不说了,有时间在『代码之谜』系列文章里面科普吧。不过有句话可以较好的概括FP和OO的区别(好吧,这个也是抄“紫皮书”上面的):

“Pascal是为了建造金字塔...Lisp是为了建造有机体...”“作为Lisp的内在数据结构,表对于这种可用性起着重要的提升作用...”“采用100函数在一个数据结构上操作,远远优于采用10个操作在十个数据结构上工作”“金字塔矗立在那里千年不变,而有机体则必须演化,否则就会消亡”。

而另一个总结得比较好的话是:(同样是抄来的)

一个对象:一组相同的运算上面,外加不同的数据。(想想你的object,是不是这样的?)

一个Closure:一组相同的数据,外加不同的操作。(Delegate就是这样的思想,有兴趣的话也可以去看看Ruby)

基本上,恩,没啥说的了。 如果你感兴趣的话,可以去看MIT SICP的课程(有在线版的,MIT也作为Open Course开设了的) 

转载于:https://my.oschina.net/justjavac/blog/91878

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值