内聚的一些理解
功能内聚性例子。比如计算雇员年龄并给出生日的子程序就是功能内聚性的,因为它只完
成一项工作,而且完成得很好。
顺序内聚性的例子。假设有一个按给出的生日计算雇员年龄、退休时间的子程序,如果它
是利用所计算的年龄来确定雇员将要退休的时间,那么它就具有顺序内聚性。而如果它是分别
计算年龄和退休时间的,但使用相同生日数据,那它就只具有通讯内聚性。
确定程序存在哪种不良内聚性,还不如确定如何把它设计得更好重要。怎样使这个子程序
成为功能内聚性呢?可以分别建立两个子程序,一个根据生日计算年龄,另外一个根据生日确
定退休时间,确定退休时间子程序将调用计算年龄的程序,这样,它们就都是功能内聚性的,
而且,其它子程序也可以调用其中任一个子程序,或这两个部调用。
通讯内聚性的例子。比如有一个打印总结报告,并在完成后重新初始化传进来的总结数据
的子程序,这个子程序具有通信内聚性,因为这两个操作仅仅是由于它们使用了相同的数据才
联系在一起。
同前例一样,我们考虑的重点还是如何把它变成是功能内聚性,总结数据应该在产生它的
地方附近被重新初始化,而不应该在打印子程序中重新初始化。把这个子程序分为两个独立的
子程序.第一个打印报告,第二个则在产生或者改动数据的代码附近重新初始化数据。然后,
利用一个较高层次的子程序来代替原来具有通讯相关的子程序,这个子程序将调用前面两个分
出来的子程序。
逻辑内聚性的例子。一个子程序将打印季度开支报告、月份开支报告和日开支报告.具体
第五章 高质量子程序的特点 55
打印哪一个,将由传入的控制标志决定,这个子程序具有逻辑内聚性,因为它的内部逻辑是由
输进去的外部控制标志决定的。显然,这个子程序不是按只完成一项工作并把它作好的原则。
怎样使这个子程序变为功能内聚性呢?建立三个子程序:一个打印季度报告,一个打印月
报告、一个打印日报告,改进原来的子程序,让它根据传送去控制标志来调用这三个子程序之
一。调用子程序将只有调用代码而没有自己的计算代码,因而具有功能内聚性。而三个被调用
的手程序也显然是功能内聚性的。非常巧合,这个只负责调用其它子程序的子程序也是一个事
务处理中心。最好用如DispatchReporPrinting()之类带有“调度”或“控制”等字眼的词来给事
务处理中心命名,以表示它只负责命令温调度,而本身并不做任何工作。
逻辑内聚性的另一个例子。考虑一个负责打印开支报表、输入新雇员名字并备份数据库的
子程序,其具体执行内容将由传入的控制标志控制。这个子程序只具有逻辑内聚性,虽然这个
关联看起来是不合逻辑的。
要想使它成为功能内聚性,只要按功能把它分成几部分就可以了。不过,这些操作有些过
于凌乱。因此,最好重新建立一个调用各子程序的代码。当拥有几个需要调用的子程序时,重
新组织调用代码是比较容易的。
过程内聚性的例子。假设有一个子程序,它产生读取雇员的名字,然后是地址,最后是它
的电话号码。这种顺序之所以重要,仅仅是因为它符合用户的要求,用户希望按这种顺序进行
屏幕输入。另外一个子程序将读取关于雇员的其它信息。这个子程序是过程内聚性,因为是由
一个特定顺序而不是其它任何原因,把这些操作组合在一起的。
与以前一样,如何把它变为功能内聚性的答案仍然是把它分为几个部分,并把这几部分分
别放入程序中。要保证调用子程序的功能是单一、完善的。调用子程序应该是诸如
GetEmployeeData()这样的子程序,而不该是像GetFirstPartofEmployeeData()这类的子程序。可能
还要改动其余读取数据的子程序。为得到功能内聚性,改动几个子程序是很正常的