打印哪一个,将由传入的控制标志决定,这个子程序具有逻辑内聚性,因为它的内部逻辑是由
输进去的外部控制标志决定的。显然,这个子程序不是按只完成一项工作并把它作好的原则。
怎样使这个子程序变为功能内聚性呢?建立三个子程序:一个打印季度报告,一个打印月
报告、一个打印日报告,改进原来的子程序,让它根据传送去控制标志来调用这三个子程序之
一。调用子程序将只有调用代码而没有自己的计算代码,因而具有功能内聚性。而三个被调用
的手程序也显然是功能内聚性的。非常巧合,这个只负责调用其它子程序的子程序也是一个事
务处理中心。最好用如DispatchReporPrinting()之类带有“调度”或“控制”等字眼的词来给事
务处理中心命名,以表示它只负责命令温调度,而本身并不做任何工作。
逻辑内聚性的另一个例子。考虑一个负责打印开支报表、输入新雇员名字并备份数据库的
子程序,其具体执行内容将由传入的控制标志控制。这个子程序只具有逻辑内聚性,虽然这个
关联看起来是不合逻辑的。
要想使它成为功能内聚性,只要按功能把它分成几部分就可以了。不过,这些操作有些过
于凌乱。因此,最好重新建立一个调用各子程序的代码。当拥有几个需要调用的子程序时,重
新组织调用代码是比较容易的。
过程内聚性的例子。假设有一个子程序,它产生读取雇员的名字,然后是地址,最后是它
的电话号码。这种顺序之所以重要,仅仅是因为它符合用户的要求,用户希望按这种顺序进行
屏幕输入。另外一个子程序将读取关于雇员的其它信息。这个子程序是过程内聚性,因为是由
一个特定顺序而不是其它任何原因,把这些操作组合在一起的。
与以前一样,如何把它变为功能内聚性的答案仍然是把它分为几个部分,并把这几部分分
别放入程序中。要保证调用子程序的功能是单一、完善的。调用子程序应该是诸如
GetEmployeeData()这样的子程序,而不该是像GetFirstPartofEmployeeData()这类的子程序。可能
还要改动其余读取数据的子程序。为得到功能内聚性,改动几个子程序是很正常的。
同时具有功能和临时内聚性的程序。考虑一个具有完成一项事物处理所有过程的子程序,
从用户那里读取确认信息,向数据存入一个记录,清除数据域,并给计数器加1。这个程序是
功能内聚性的,因为它只从事一项工作,进行事物处理,但是,更确切地说,这个子程序同时
也是临时内聚性的,不过当一个子程序具有两种以上内聚性时,一般只考虑最强的内聚性。
这个例子提出了如何用一个名字恰如其分地抽象描述出程序内容的问题。比如可以称这
个子程序为ConfirmEntryAndAdjustData(),表示这个干程序仅具有偶然内聚性。而如果称它为
CompleteTransaction(),那么就可能清楚地表示出这个子程序仅具有一个功能,而已具有功能内
聚性。
过程性、临时或者可能的逻辑内聚性。比如一个进行某种复杂计算前5个操作,并把中间
结果返回到调用子程序。由于5 项操作可能要用好几个小时,因此当系统瘫痪时,这个子程序
将把中间结果存入一个文件中,然后,这个子程序检查磁盘,以确定其是否有足够空间来存储
最后计算结果,并把磁盘状态和中间结果返回到调用程序。
这个子程序很可能是过程内聚性的,但你也可能认为它具有临时内聚性,甚至具有逻辑内
聚性。不过,不要忘了问题的关键不是争论它具有哪种不好的内聚性,而是如何改善其内聚性。
原来的子程序是由一系列令人莫名其妙的操作组成的,与功能内聚性相距甚远,首先,调
用子程序不应该调用一个,而应该调用几个独立的子程序:l)进行前5步计算的子程序;2)把
中间结果存入一个文件;3)确定可用的磁盘存储空间。如果调用子程序被称作