你的编程语言可以这样做吗?


Posted in Javascript onSeptember 07, 2006

有一天,你在浏览自己的代码,发现有两大段代码几乎一样。实际上,它们确实是一样的——除了一个关于意大利面(Spaghetti)而另一个关于巧克力慕思(Chocolate Moose)。 

  // 一个小例子:

  alert("偶要吃意大利面!");
  alert("偶要吃巧克力慕思!");
嗯,这个例子碰巧是用javascript写的,不过你就算不懂JavaScript,应该也能明白它在干什么。 

拷贝代码不好。于是,你创建了个函数 

  function SwedishChef( food ){
      alert("偶要吃" + food + "!");
  }
  SwedishChef("意大利面");
  SwedishChef("巧克力慕思");
Ok,这只是一个很小很小的例子而已,相信你能想像到个更实际一点的例子。这段代码有很多优点,你全都听过几万次了:可维护性、可读性、抽象性 = 好! 

现在你留意到有另外两段代码几乎跟它们一模一样,除了一个反复调用一个叫BoomBoom的函数,另一个反复调用一个叫PutInPot的。除此之外,?两段代码简直没什么两样: 

  alert("拿龙虾");
  PutInPot("龙虾");
  PutInPot("水");
  alert("拿鸡肉");
  BoomBoom("鸡肉");
  BoomBoom("椰子酱");
现在要想个办法,使得你可以?⒁桓龊??米髁硪桓龊??牟问?U馐歉鲋匾?哪芰Γ?蛭?愀?菀捉?蚣艽?胄闯梢桓龊???mu注:还记得template method模式吧?)。 

  function Cook( i1, i2, f ){
      alert("拿" + i1);
      f(i1);
      f(i2);
  }
  Cook( "龙虾", "水", PutInPot );
  Cook( "鸡肉", "椰子酱", BoomBoom );
看看,我们居然把函数当成调用参数传递了! 

你的编程语言能办到吗? 

等等……假如我们已经有了PutInPot和BoomBoom这些函数的具体实现代码(而且又不需要在别的地方重用它们),那么用内联语法把它们写进函数调用里面不是比显式的声明这两个函数更漂亮吗? 

  Cook( "龙虾", 
        "水", 
        function(x) { alert("pot " + x); }  );
  Cook( "鸡肉", 
        "椰子酱", 
        function(x) { alert("boom " + x); } );
耶,真方便!请注意我只是随手创建了个函数,甚至不用考虑怎么为它起名,只要拎着它的耳朵把它往一个函数里头一丢就可以了。
当你一想到作为参数的匿名函数,你也许想到对那些对数组里的每个元素进行相同操作的代码。 

  var a = [1,2,3];
  for (i=0; i<a.length; i++){
      a[i] = a[i] * 2;
  }
  for (i=0; i<a.length; i++){
      alert(a[i]);
  }
常常要对数组里的所有元素做同一件事,因此你可以写个这样的函数来帮忙: 

  function map(fn, a){
      for (i = 0; i < a.length; i++){
          a[i] = fn(a[i]);
      }
  }
现在你可以把上面的东西改成: 

  map( function(x){return x*2;}, a );
  map( alert, a );
另一个常见的任务是将数组内的所有元素按照某总方式汇总起来: 

  function sum(a){
      var s = 0;
      for (i = 0; i < a.length; i++)
          s += a[i];
      return s;
  }

  function join(a){
      var s = "";
      for (i = 0; i < a.length; i++)
          s += a[i];
      return s;
  }

  alert(sum([1,2,3]));
  alert(join(["a","b","c"]));
sum和join长得很像,你也许想把它们抽象为一个将数组内的所有元素按某种算法汇总起?淼姆盒秃???nbsp;

  function reduce(fn, a, init){
      var s = init;
      for (i = 0; i < a.length; i++)
          s = fn( s, a[i] );
      return s;
  }

  function sum(a){
      return reduce( function(a, b){ return a + b; }, a, 0 );
  }

  function join(a){
      return reduce( function(a, b){ return a + b; }, a, "" );
  }
许多早期的编程语言没法子做这种事。有些语言容许你做,却又困难重重(例如C有函数指针,但你要在?e处声明和定义函数)。面向对象语言也不确保你用函数可以干些啥(把函数当对象处理?)。 

如果你想将函数视为一类对象,Java要求你建立一个有单方法的对象,称为算子对象。许多面向对象语言要你为每个类都建立一个完整文件,像这样开发可真叫快。如果你的编程?言要你使用算子对象来包装方法(而不是把方法本身当成对象),你就不能?氐椎玫较执?ǘ??┍喑逃镅缘暮么Α2环潦允钥茨憧煞裢嘶跄没匦┣??nbsp;

不用再写那些除了经过一个数组对每个元素做一些事情之外一无是处的函数,有什么好处? 

让我们看回map函数。当你要对数组内的每个元素做一些事,你很可能不在乎哪个元素先做。无论由第一个元素开始执行,还是是由最后一个元素执行,你的结果都是一样的,对不?如果你手头上有2??CPU,你可以写段代码,使得它们各对一半的元素工作,于是乎map快了两倍。 

或者,发挥一下想像力,设想你在全球有千千万万台服务器分布在全世界的若干个数据中心,你有一个真的很大很大的数组,嗯,再发挥一下想像力,设想这个数组记录有整个互联网的内容。还了,现在你可以在几千台服务器上同时执行map,让每台服务器都来解决同一个问题的一小部分。 

那么在这个例子里面,编写一段非常快的代码来搜索整个互联网这个问题,其实就和用一个简单的字符串搜索器(算子)作为参数来调用map函数一样简单了。 

希望你注意到一个真正有意思的要点,如果你想要把map/reduce模式变成一个对所有人都有用,对所有人都能立刻派上用场的技术,你只需要一个超级天才来写最重要的一部分代码,来让map/reduce可以在一个巨大的并行计算机阵列上运行,然后其他旧的但是一向在单一个循环中运行良好的代码,仍可以保持正确的运行,惟一的差别只是比原来单机运行快了n倍。这意味着它们都一不留神突然变成可以被用来解决一个巨大的问题的代码。 

让我再??乱幌拢?ü?选把?贰闭飧龈拍罴右猿橄螅?憧梢园延萌魏文阆不兜姆绞嚼词迪帧把?贰惫?蹋??ǹ梢允迪秩醚?返??俣人孀庞布?扑隳芰Ρ3至钊寺?獾耐?皆龀ぁ?nbsp;

你现在应该可以明白不久为何对那些对除了Java之外什么都?]被学过的计算机系学生表示不满了:( http://www.joelonsoftware.com/articles/ThePerilsofJavaSchools.html) :

Without understanding functional programming, you can't invent MapReduce, the algorithm that makes Google so massively scalable. The terms Map and Reduce come from Lisp and functional programming. MapReduce is, in retrospect, obvious to anyone who remembers from their 6.001-equivalent programming class that purely functional programs have no side effects and are thus trivially parallelizable. The very fact that Google invented MapReduce, and Microsoft didn't, says something about why Microsoft is still playing catch up trying to get basic search features to work, while Google has moved on to the next problem: building Skynet^H^H^H^H^H^H the world's largest massively parallel supercomputer. I don't think Microsoft completely understands just how far behind they are on that wave. 

不理解函数式编程,你就发明不了MapReduce这个让Google的计算能力如此具有可扩展性的算法。Map和Reduce这两个术语源自Lisp语言和函数式编程……(这是另一篇文章的内容,emu也不是很理解其中的各种说法的来龙去脉,就不翻译了) 

我希望你现在明白,把函数当成基本类型的(动态)编程语言能让你在编程过程中更好的进行抽象化,也就是使代码精悍、功能更内聚、更具可重用性及更具有扩展性。很多的Google应用使用Map/Reduce模式,因此一有人对其优化或修正缺陷,它们就都可以从中得益。 

我准备要再罗嗦一下,我认为最有生产力的编程语言莫过于能让你在不同层次上都可以进行抽象化的。老掉牙的FORTRAN 语言以前是不让你写函数的注。C 有函数指针,可是它们都非常丑丑丑丑丑丑丑丑陋,不允许匿名声明,又不能在用它们时实现它们而偏偏要放在?e处去实现。Java让你使用算子对象,一种更丑陋的东西。正如Steve Yegge所述,Java是??名词王国 (http://steveyegge.blogspot.com/2006/03/execution-in-kingdom-of-nouns.html)。 

作者注:这里提起了FORTRAN,不过我上次使用FORTRAN是27年前的事了。FORTRAN是有函数的,我码字那会儿脑子里面想的大概是GW-BASIC语言。(emu注,basic确实只有所谓的子程序和go-sub语句,作用只是重新组织代码结构而已,没有参数和调用堆栈,因此没有真正的函数调用)

译者注:原作者起了《你的编程语言可以这样做吗》这个标题其实并不是这篇文章的真正价值所在,我转这篇文章也不是因为原作者可以把语言的初级技巧玩得转,而是因为这是一篇map/reduce模型的示范。

Javascript 相关文章推荐
javascript 触发HTML元素绑定的函数
Sep 11 Javascript
js将字符串转成正则表达式的实现方法
Nov 13 Javascript
javascript函数定义的几种区别小结
Jan 06 Javascript
C++中的string类的用法小结
Aug 07 Javascript
基于javascript实现图片滑动效果
May 07 Javascript
JS输出空格的简单实现方法
Sep 08 Javascript
详解vue 数组和对象渲染问题
Sep 21 Javascript
详解Express笔记之动态渲染HTML(新手入坑)
Dec 13 Javascript
Vue路由对象属性 .meta $route.matched详解
Nov 04 Javascript
vue+elementUI(el-upload)图片压缩,默认同比例压缩操作
Aug 10 Javascript
Vue如何将页面导出成PDF文件
Aug 17 Javascript
微信小程序基于高德地图API实现天气组件(动态效果)
Oct 22 Javascript
音乐播放用的的几个函数
Sep 07 #Javascript
在网页中屏蔽快捷键
Sep 06 #Javascript
js的event详解。
Sep 06 #Javascript
农历与西历对照
Sep 06 #Javascript
MSN消息提示类
Sep 05 #Javascript
经验几则 推荐
Sep 05 #Javascript
JS实现浏览器菜单命令
Sep 05 #Javascript
You might like
PHP+DBM的同学录程序(1)
2006/10/09 PHP
随时给自己贴的图片加文字的php代码
2007/03/08 PHP
解析二进制流接口应用实例 pack、unpack、ord 函数使用方法
2013/06/18 PHP
php用户登录之cookie信息安全分析
2016/05/13 PHP
Laravel使用消息队列需要注意的一些问题
2017/12/13 PHP
在TP5数据库中四个字段实现无限分类的示例
2019/10/18 PHP
WebGame《逆转裁判》完整版 代码下载(1月24日更新)
2007/01/29 Javascript
IE/FireFox具备兼容性的拖动代码
2007/08/13 Javascript
jquery.boxy插件的iframe扩展代码
2010/07/02 Javascript
jQuery性能优化28条建议你值得借鉴
2013/02/16 Javascript
js函数返回多个返回值的示例代码
2013/11/05 Javascript
js获取checkbox值的方法
2015/01/28 Javascript
Sublime Text 3常用插件及安装方法
2015/12/16 Javascript
实例详解JavaScript中setTimeout函数的执行顺序
2017/07/12 Javascript
利用canvas中toDataURL()将图片转为dataURL(base64)的方法详解
2017/11/20 Javascript
vue数据控制视图源码解析
2018/03/28 Javascript
解决vue2.0 element-ui中el-upload的before-upload方法返回false时submit()不生效问题
2018/08/24 Javascript
解决Vue2.0 watch对象属性变化监听不到的问题
2018/09/11 Javascript
详解js模板引擎art template数组渲染的方法
2018/10/09 Javascript
微信小程序仿抖音视频之整屏上下切换功能的实现代码
2020/05/24 Javascript
[01:28:56]2014 DOTA2华西杯精英邀请赛 5 24 CIS VS DK
2014/05/26 DOTA
python字典排序实例详解
2015/05/20 Python
解决python3 urllib 链接中有中文的问题
2018/07/16 Python
pymysql 开启调试模式的实现
2019/09/24 Python
浅谈django框架集成swagger以及自定义参数问题
2020/07/07 Python
英国顶级家庭折扣店:The Works
2017/09/06 全球购物
英国最好的包装供应商:Priory Direct
2019/12/17 全球购物
Herschel Supply Co.美国:背包、手提袋及配件
2020/11/24 全球购物
物理教师自荐信范文
2013/12/28 职场文书
会员活动策划方案
2014/08/19 职场文书
违反交通法规检讨书
2014/09/10 职场文书
春季运动会开幕词
2015/01/28 职场文书
保研导师推荐信
2015/03/25 职场文书
护理心得体会范文
2016/01/22 职场文书
mysql使用 not int 子查询隐含陷阱
2022/04/12 MySQL
Win11显卡控制面板打开显卡设置方法
2022/04/20 数码科技