杂谈现代高级编程语言

标签: Uncategorized | 发表时间:2011-06-19 13:14 | 作者:cxwangyi 卡飞菲
出处:http://cxwangyi.wordpress.com

几个月之前,Slashdot转载了Robert Harper教授的一篇博客,说卡内基梅隆大学计算机系把”面向对象编程“从大一新生的必修课中删掉了,其原因是

Object-oriented programming … is both anti-modular and anti-parallel by its very nature.

这两个原因(anti-modular和anti-parallel)都是很重的指责了;尤其是anti-modular,因为OO的基本思想通常被理解成“封装”,从而实现模块化。

我是在1995年第一次听说“面向对象”(Object Oriented)这个说法。当时在学习正在成长过程中的C++,用的是Borland C++ 1.0。从那时开始的很多年里,”类“(class)、“对象”(object)和“方法”(methods),以及在这些概念之上构建的”继承“(inheritance)和”多态“(polymorphism)都是我理解中OO最核心的思想。我猜大多数程序员在这方面的认识都和我差不多。

但是“封装”真的是OO的本质嘛?直到最近为了给iPhone写个玩具程序而学习Objective-C(一种非常古老和原始的面向对象编程语言)的时候,才注意到早在1998年,OO之父Alan Kay就曾经在一篇邮件中说,他很后悔发明了“object”这个词,从而误导大家,把注意力都集中到“封装”,而忽视了OO的本质——messaging(消息传递)。Alan Kay的原话是:

The big idea is “messaging” … . The key in making great and growable systems is much more to design how its modules communicate rather than what their internal properties and behaviors should be.

Objective-C的设计是非常强调“消息传递”(messaging)的——对一个object的method的调用,被称为“给这个object发了一个消息”。为了突出调用method时指定的参数(parameters)实际上是消息中的一些内容,Objective-C不惜把method的定义方式都做了相对于C的很大的修改,从而把参数嵌入在method的名字里。比如在一个叫做myWebView对象中搜索一段文字,要求不区分大小写,从前往后搜索,用Objective-C来描述是:

[myWebView searchFor:myString 
           direction:YES
       caseSensitive:NO
                wrap:YES]

而用C++或者Java来描述,则是

myWebView.searchFor(myString, YES, NO, YES)

乍看上去,C++ 或者Java的方式更简短,但是Objective-C的方式更强调“发消息”。实际上,上面Objective-C语句会被翻译成如下C函数调用:

objc_msgSend(myWebView,
             searchFor:direction:caseSensitive:wrap:,
             myString, YES, NO, YES)

从强调messaging的角度看,Objective-C确实比C++和Java更符合Alan Kay对OO思想的描述。

OO中的messaging思想不仅体现在Objective-C语言以及在其上构建的NextSTEP/Cocoa GUI编程套件上。在Cocoa因为Mac OS X和iPhone流行起来之前,很多人都接触过Qt(一种基于C++语言的GUI开发套件)。Qt对messaing的支持比Objective-C/Cocoa更彻底——每个object可以发出若干signal,每个signal可以触发这个object自己的或者其他objects的若干个slot。

有意思的是,为了支持messaging,Qt对C++语言做了扩展,而Objective-C对C语言做了扩展。这两套扩展都利用了起源于C语言的“宏”机制(macro)。类似的做法也可以用于Java,前提是我们在调用Java编译器之前,先调用一下cpp宏展开程序来预处理一下我们的Java程序。这事儿可以留待Java爱好者们来搞?

不管是Objective-C还是Qt,都会“尽力”去检查一个object是否支持一个method(或者叫message),但是并不禁止程序员向一个object发送一个它不认识的message(或者调用一个object没有的method)。说“尽力”检查,是因为两者都不能保证检查的完备性。这是因为Objective-C和Qt都支持多态;具体的说,接受message的object可能是表示为一个指针(指向object),所以直到运行时候,当一个message抵达某个object的时候,系统才能(通过查这个object对应的message list)知道这个object是否认识这个message。

这种灵活性在Google新推出的Go语言中也同样实现了,而且做的很极致——Go语言中没有class的概念;换句话说,不需要是class类型的object才能有对应的方法(methods)——Go允许给几乎任何类型附上methods。而且程序中可以很方便的检测一个object是否支持(一组)methods,比如:

type Stringer interface {
        String() string
}
s, ok := v.(Stringer)  // Test whether object v implements "String()"

【和Go的这种灵活性类似的,Objective-C允许给已经定义了的class增加一些methods,而不需要derive subclass;Objective-C的这种机制被称为category。和Go不同的是,一个category是对某个class的一个扩展,而Go语言里完全没有class了(但是有interface的概念)。】

说到这里,我觉得差不多可以反过来理解Robert Harper教授对OO的评价了——其实Robert不是在藐视OO,而是在指责很多imperative OO languages(我理解包括Java和未经Qt扩展的C++;详见后述),认为这些语言没有完成实现OO中object messaging的核心思想,从而不算实现了“模块化“(modulization)的思想。

上述都是关于程序的模块化。实际上,模块化的另一个主要方面是对“数据”(data)的模块化。从图灵机和lambda-calculus开始,计算机科学家们就注意到程序和数据是统一的;比如在冯诺依曼的“二进制存储电子计算机”模型里,程序和数据都是bit stream。即时我们在讨论高级编程语言的时候,程序和数据也不应该被分开。因为现代数据操作和模块化的基础是并行程序(parallelism),而有效实现并行的基础是程序的first-class表达,也就是把程序作为一种基本数据类型。

鉴于这篇帖子已经很长了,这段话就作为下一篇帖子的提纲吧。下一篇帖子里,我们来说说XML、JSON、MessagePack、Protocol Buffers这些persistent data structure,以及用源于古老的functional programming paradigm的Go语言和MapReduce实现的并行数据操作。


相关 [现代 编程语言] 推荐:

杂谈现代高级编程语言

- 卡飞菲 - Yi Wang's Tech Notes
几个月之前,Slashdot转载了Robert Harper教授的一篇博客,说卡内基梅隆大学计算机系把”面向对象编程“从大一新生的必修课中删掉了,其原因是. 这两个原因(anti-modular和anti-parallel)都是很重的指责了;尤其是anti-modular,因为OO的基本思想通常被理解成“封装”,从而实现模块化.

[转][转]Clojure 编程语言

- - heiyeluren的blog(黑夜路人的开源世界)
简介: Lisp 是一种编程语言,以表达性和功能强大著称,但人们通常认为它不太适合应用于一般情况. Clojure 是一种运行在 Java™ 平台上的 Lisp 方言,它的出现彻底改变了这一现状. 如今,在任何具备 Java 虚拟机的地方,您都可以利用 Lisp 的强大功能. 在本文中,了解如何开始使用 Clojure,学习它的一些语法,同时利用 Eclipse 的 Clojure 插件提供帮助.

如果编程语言是汽车……

- 康爷 - Wow! Ubuntu
原文: If programming languages were cars... http://www.cs.caltech.edu/~mvanier/hacking/rants/cars.html,有筛减. 本文是一系列把编程语言和汽车对比的老笑话合集,添加了几种最近流行的语言. 注:似乎 Digg/Reddit/StumbleUpon 对本文进行了挖掘/阅读/踩,结果我收到了一大坨儿的新建议.

如果编程语言是一条船…

- will - cnBeta.COM
Turing绝对是一个双人皮划艇.

你喜欢什么编程语言?

- Sean - 不周山
OpenParty回来,重看了一下自己的讲稿,觉得还是挺乱的,想传递的东西很多,仓促之间也没有把内容组织好,对周六冒着桑拿天赶到现场,还在一间没有空调的挤了一百多号人的房间里听我唠叨了近一个小时的同学们深感有愧. 这里摘要一下,也许能同时弥补一下因为我口音不纯带来的表意不清. 其实主要是两部分的内容,一是我认为R能极大提高算法与数据工作人员的快乐程度及工作效率;二是R的发展历史中呈现出来的六张面孔.

年度编程语言锁定Objective-C

- ndv - 《程序员》杂志官网
虽然2011年才刚过半,但随着iPhone和iPad平台市场领导地位的确立和Apple各系统不断推陈出新,从市场占有的趋势来看,Objective-C成为年度编程语言无疑. TIOBE编程语言排行榜每年初都将统计评出上年度的最佳编程语言大奖,获奖的编程语言被认为在该年度市场份额增长及用户认可度方面获得了最为突出的成绩.

Google将宣布新编程语言Dart

- 李龑 - Solidot
Google的两位工程师将在下个月举行的Goto软件开发会议上宣布一门全新的编程语言Dart. 这门新语言是为结构化的Web编程设计的,适用于开发基于浏览器的Web应用. Google还没有透露Dart的任何技术细节. Google在2009年曾发布了主要针对服务器端的Go语言.

编程语言之间的界限

- - ITeye资讯频道
Cory House是一个软件开发者,笃信清洁代码和务实开发. 最近他总结了不同的编程语言之间存在的界限规则,他还认为不了解编程语言之间存在的界限无异于选择错误的开发工具,要想简化、加快软件开发过程,了解语言之间的异同是必不可少的. 如今开发者有很多种途径去解决编程问题,不管是利用大量的类库还是各种各样的技术工具.

异步编程语言的常见坑

- - idea's blog
天生支持异步编程的语言如 NodeJS, Golang 等, 创建一个异步 routine 的成本非常小, 这确实是一个非常方便的功能. 比如用在网络爬虫程序的开发, 对于每一个要抓取的 URL 就启动一个 routine, 类似启动一个线程, 既能充分利用 CPU 多核, 代码也很简洁.. 正因为太方便, 所以常常被滥用, 并引发许多严重坑.

Peter Norvig:编程语言的选择并不重要

- jeff - python.cn(jobs, news)
Lisp被许多资深程序员视为编程语言中的圣杯,因为学起来很难. 著名程序员、最热门的技术问答网站StackOverflow创始人Joel Spolsky曾经在“Java语言学校的危险性”一文中说,自己当年在大. 学里学习用Lisp的导论课程也是苦不堪言,而Lisp这样的函数式语言实际开发中并不常用,但是一旦到了体现优势的时候,你如果不懂,将失之千里.