友情提示:如果本网页打开太慢或显示不完整,请尝试鼠标右键“刷新”本网页!
软件工程实践者的思想(PDF格式)-第3部分
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部! 如果本书没有阅读完,想下次继续接着阅读,可使用上方 "收藏到我的浏览器" 功能 和 "加入书签" 功能!
今胜昨倍许,明胜今倍许,而山不加增,何苦而不快。”
但是越发的勤快,愚公将越发没有机会找到更快的方法,
人的精力终归是有极限的。提出新的“方法”,解决
…15
…………………………………………………………Page 20……………………………………………………………
第 2 章 是懒人造就了方法
的将是影响做事成效的根本问题。而愚公可以多吃点饭,
多加点班,但突破不了人的精力的极限。
记住,在两千年前的某一天,闲极无聊的李冰下厨给
夫人炒了一个小菜,他突然发现垒灶的鹅卵石被烧得爆裂
开来,遇水尤甚。从此《史记》上记下了“蜀守冰凿离堆”,
而《华阳国志》上记下了他做这件事的方法“积薪烧之”。
在差不多同一时间,愚公在山北之塞“碎石击壤”。
2。 一百万行代码是可以写在一个文件里的
早期写程序,都是将代码打在穿孔纸带上,让计算机
去读的。要让计算机读的纸带当然是连续的,这无需多讲。
其实我也没有那样写过程序,真实的苦楚我也不知道。
…16
…………………………………………………………Page 21……………………………………………………………
『大道至简』
后来有了汇编语言,可以写一些代码了。这时的代码
是写在文本文件里,然后交给一个编译器去编译,再由一
个链接器去链接,这样就出来了程序。
第一个写汇编的人,可能写的是有名的“Hello World ”
程序,那个程序写在一个文件里就行了。所以后来就成了
习惯,大家都把代码写到一个文件里。早期的汇编语言里,
GOTO 语句是用得非常非常频繁的,将一个语句 GOTO
到另一个文本文件里去,既不现实也不方便。所以大家习
以为常,便统统地把代码写到一个文件里。
再后来出了高级语言,什么 C 呀,Pascal 呀之类的。
既然大家已经形成习惯了,所以很自然地会把一个程序写
到一个文件里。无论这个程序有多大,多少行代码,写到
一个文件里多方便呀。
直到如今语言发展得更高级了。可是程序员的习惯还
是难改,一旦得了机会,还是喜欢把代码写到一个文件里
的。
好了,有人说我是想当然尔。En ,这当然是有实据的。
记得 Delphi 1。0 版发布的时候,全世界一片叫好声。连“不
支持双字节”这样的大问题,都不影响他在华语地区的推
广。然而不久,爆出了一个大 BUG !什么大 BUG 呢?
Delphi 1。0 的编译器居然不支持超过 64K 的源代码文件!
这被 Fans 们一通好骂。直到我用 Delphi 2。0 时,一
个从 VB 阵营转过来的程序员还跑过来问我这件事。好在
Delphi 2。0 改了这个 BUG ,这让当时我的面子上好一阵风
…17
…………………………………………………………Page 22……………………………………………………………
第 2 章 是懒人造就了方法
光。
64k 的文件是什么概念呢?
1 行代码大概(平均)是 30 字节,64k 的源代码是 2184
行,如果代码风格好一点,再多一些空行的话,差不多也
就是 3000 行上下。
也就是说,在 Delphi 1 的时代( 以及其后的很多很多
时代) ,程序员把 3000 行代码写到一个文件里,是司空见
惯的事了。如果你不让他这样写,还是会被痛骂的呢。
所以呢,按照这一部分人的逻辑,一百万行代码其实
是可以写在一个文件里的。不单可以,而且编译器、编辑
器等等也都必须要支持。这才是正统的软件开发。
勤快的愚公创造不了方法。这我已经说过了。对于要
把“一百万行代码写到一个文件”,查找一个函数要在编
辑器里按五千次 PageDown/PageUp 键的勤快人来说,是
不能指望他们创造出“单元文件(Unit) ”这样的开发方法
来的。
然而单元文件毕竟还是出现了。这个世界上,有勤快
人就必然有懒人,有懒人也就必然有懒人的懒方法。
有了单元文件,也就很快出现了一个新的概念:模块。
把一个大模块分成小模块,再把小模块分成更细的小小模
块,一个模块对应于一个单元。于是我们可以开始分工作
…18
…………………………………………………………Page 23……………………………………………………………
『大道至简』
了,一部分人写这几个单元的代码,另一部分则写那几个。
很好,终于可以让源代码分散开来。结构化编程的时
代终于开始了,新的方法取代了旧的方法,而这一切的功
劳,是要归终于那个在按第 5001 次PageDown键时,突然
崩溃的程序师。他发自良心地说:不能让这一切继续下去
了,我一定要把下一行代码写到第二个文件里去。我发誓,
我要在编译器里加入一个Unit 关键字。①
3。 你桌上的书是乱的吗?
几周之前,在一所电脑培训学校与学生座谈时,一个
学员问我:“为什么我学了一年的编程,却还是不知道怎
么写程序呢”。
我想了想,问了这个学员一个问题:“你桌上的书是
乱的吗?”
他迟疑了一下,不过还是回答我道:“比较整齐。”
我当时便反问他:“你既然知道如何把书分类、归整
得整整齐齐地放在书桌,那怎么没想过如何把所学的知道
分类一下,归纳一下,整整齐齐地放在脑子里呢?”
如果一个人学了一年的编程,他的脑袋里还是昏乎乎
的,不知道从哪里开始,也不知道如何做程序。那想来只
① Turbo Pascal 3。0 中才开始有了Uses 和Unit 关键字。在ANSI Pascal
标准里并没有它。
…19
…………………………………………………………Page 24……………………………………………………………
第 2 章 是懒人造就了方法
有一个原因:他学了,也把知识学进去了,就是不知道这
些知识是干什么的。或者说,他不知道各种知识都可以用
来做什么。
其实结构化编程的基本单位是“过程(Procedure) ”,
而不是上一小节说到的“单元(Unit) ”。然而在我看来,过
程及其调用是 CPU 指令集所提供的执行逻辑,而不是普
通的开发人员在编程实践中所总结和创生的“方法”。
这里要提及到CPU指令集的产生。产生最初的指令集
的方式我已经不可考证,我所知道的是CISC 指令集与
RISC 指令集之争在 1979 年终于爆发。前者被称为复杂指
令集,然而经过Patterson 等科学家的研究,发现 80% 的
CISC指令只有在 20% 的时间内才会用到;更进一步的研
究发现,在最常用的 10 条指令中,包含的流程控制只有
“条件分支(IF。。。THEN。。。)① ”、“跳转(JUMP) ” 和“调用返
回(CALL/RET) ”……
于是 CISC 被 RISC(精简指令集计算机) 替代了。动摇
CISC 指令集地位的方法,就是分类统计。
正如 CISC 指令集搅乱了一代程序设计师的思路一
样,大量的知识和资讯搅乱了上面给我提问的那位学员的
思想。他应该尝试一下分类,把既有的知识象桌子上的书
一样整理一下,最常用的放在手边,而最不常用的放在书
① 在x86 系统中,循环是用条件分支来实现的,而且条件分支指令
并不是IF。。。THEN。。。 ,这里用这两个关键字,仅用于说明问题。
…20
…………………………………………………………Page 25……………………………………………………………
『大道至简』
柜里。如果这样的话,我想他已经在九个月前就开始写第
一个软件产品了。
你桌上的书还是乱的吗?
4。 我的第一次思考:程序 = 算法 + 结构 + 方法
我的第一次关于程序的本质的思考其实发生在不久
前。那是我在 OICQ 上与 Soul 的一次谈话。
Soul(王昊)是DelphiBBS现任的总版主,是我很敬重的
一位程序员。那时我们正在做DelphiBBS 的一个“B 计划
II ”,也就是出第二本书。他当时在写一篇有关“面向对
象(OOP) ”的文章。而我正在写《Delphi源代码分析》,在
初期的版本里,有“面向对象”这一部分的内容。我们的
①
对话摘要如下 :
Soul:我在写书讨论“面向对象的局限性”
我 :En。这个倒与我的意见一致。哈哈哈。
我 :“绝对可以用面向过程的方法来实现任意复杂的系统。要知道,航
天飞机也是在面向过程的时代上的天。但是,为了使一切变得不是
那么复杂,还是出现了‘面向对象程序设计’的方法。”
我 :——哈,我那本书里,在“面向对象”一部分前的引文中。就是这
样写的。
① 这段对话的确很长。如果你不是非常有经验的程序员,那么不
能完整地阅读和理解这段文字是很正常的。部分读者甚至可以跳
过这段引文,直接阅读后面的结论。而有兴趣的读者,可以在我
的网站上读到它的全文(http://doany/) 。
…21
…………………………………………………………Page 26……………………………………………………………
第 2 章 是懒人造就了方法
Soul:现在的程序是按照冯。诺伊曼的第一种方案做的,本来就是顺序的,
而不是同步的。CPU怎么说都是一条指令一条指令执行的。
我 :面向过程是对“流程”、“结构”和“编程方法”的高度概括。而
面向对象本身只解决了“结构”和“编程方法”的问题,而并没有
对“流程”加以改造。
Soul:确实如此。确实如此。
我 :对流程进一步概括的,是“事件驱动”程序模型。而这个模型不是
OO提出的,而是Windows的消息系统内置的。所以,现在很多人迷
惑于“对象”和“事件”,试图通过OO来解决一切的想法原本就是
很可笑的。
Soul:我先停下来,和你讨论这个问题,顺便补充到书里去。
我 :如果要了解事件驱动的本质,就应该追溯到Windows内核。这样就
涉及到线程、进程和窗体消息系统这些与OO无关的内容。所以,整
个RAD的编程模型是OO与OS一起构建的。现在很多的开发人员只知
其OO的外表,而看不到OS的内核,所以也就总是难以提高。
Soul:OO里面我觉得事件的概念是很牵强的,因为真正的对象之间是相互
作用;就好像作用力和反作用力;不会有个“顺序”的延时。
我 :应该留意到,整个的“事件”模型都是以“记录”和“消息”的方
式来传递的。也就是说,事件模型停留在“面向过程”编程时代使
用的“数据结构”的层面上。因此,也就不难明白,使用/不使用
OO都能写Windows程序。
我 :因为流程还是在“面向过程”时代。
Soul:所以所谓的面向对象的事件还是“顺序”的。所以我们经常要考虑
一个事件发生后对其他过程的影响,所以面向对象现在而言是牵强
的。
我 :如果你深入OS来看SEH,来看Messages,就知道这些东西原本就不
是为了“面向对象”而准备的。面向对象封装了这些,却无法改造
它们的流程和内核。因为OO的抽象层面并不是这个。
我 :事件的连续性并不是某种编程方法或者程序逻辑结构所决定的。正
如你前面所说的,那是CPU决定的事。
Soul:比如条件选择,其实也可以用一种对象来实现,而事实没有。这个
是因为cpu的特性和面向对象太麻烦。
我 :可能,将CPU做成面向对象的可能还是比较难于想象和理解。所以
MS才启动 Framework。我不认为在面向对象方法上有什么
…22
…………………………………………………………Page 27……………………………………………………………
『大道至简』
超越,也不认为它的FCL库会有什么奇特的地方。——除了它们足
够庞大。但是我认为,如果有一天OS也是用 Framework来编写
的,OS一级的消息系统、异常机制、线程机制等等都是的,都
是面向对象的。那么,在这个基础上,将“事件驱动”并入OO层面
的模型,才有可能。
Soul:所以我发觉面向对象的思维第一不可能彻底,第二只能用在总体分
析层上。在很多时候,实质上我们只是把一个顺序的流程折叠成对
象。
我 :倒也不是不可能彻底。有绝对OO的模型,这样的模型我见过。哈
哈~~但说实在的,我觉得小应用用“绝对OO”的方式来编写,有
失“应用”的本意。我们做东西只是要“用”,而不是研究它用的
是什么模型。所以,“Hello World”也用OO方式实现,原本就只
是出
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部!
温馨提示: 温看小说的同时发表评论,说出自己的看法和其它小伙伴们分享也不错哦!发表书评还可以获得积分和经验奖励,认真写原创书评 被采纳为精评可以获得大量金币、积分和经验奖励哦!