Summary:若干客户使用类接口中的同一子集,或者两个类的接口有相同部分。将相同的子集提炼到一个独立接口中。
动机:
类之间彼此相互用的方式有若干种。“使用一个类”通常意味用到该类的所有责任区。另一种情况是,某一组客户只使用类责任区中的一个特定子集。再一种情况则是,这个类需要与所有协助处理某些特定请求的类合作。
对于后两种情况,将真正用到的这部分责任分离出来通常很有意义,因为这样可以使系统的用法更清晰,同时也更容易看清系统的责任划分。如果新的类需要支持上述子集,也比较能够看清子集内有些什么东西。
在许多面向对象语言中,这种责任划分是通过多继承(multiple inheritance)来实现的。你可以针对每组行为建立一个类,再将它们组合于同一个实现中。Java只提供单继承(single inheritance),但你可以运用接口来昭示并事项上述需求。接口对于java 程序的设计方式有着巨大的影响,就连Smalltalk程序员都认为接口是一大进步。
Extract Superclass 和Extract Interface之间有些相似之处。Extract Interface 只能提炼共通接口,不能提炼共通代码。使用Extract Interface可能造成难闻的“重复”坏味道,幸而你可以运用Extract Class 先把共通行为放进一个组件中,然后将工作委托该组件,从而解决这个问题。如果有不少共通行为,Extract Superclass会比较简单,但是每个类只能有一个超类。
如果某个类在不同环境下扮演截然不同的角色,使用接口就是个好主意。你可以针对每个角色以Extract Interface提炼出相应接口。另一种可以用上Extract Interface的情况是:你想要描述一个类的外部依赖接口(outbound interface,即这个类要求服务提供方提供的操作)。如果你打算将来加入其他种类的服务对象,只需要求它们实现这个接口即可。
做法:
1.新建一个空接口
2.在接口中声明待提炼类的共通操作。
3.让相关的类实现上述接口
4.调整客户端的类型声明,令其使用该接口。
范例:
TimeSheet类表示员工为客户工作的时间表,从中可以计算客户应该支付的费用。为了计算这笔费用,TimeSheet需要知道员工级别,以及员工是否有特殊技能
double charge(Employee emp,int days){int base = emp.getRate() * days;if(emp.hasSpecialSkill()){return base * 1.05;}else{return base;}
}
除了提供员工的级别和特殊技能信息外,Employee还有很多其他方面的功能,但本应用程序只需这两项功能。我们可以针对这两项功能定义一个接口,从而强调“我只需要这部分功能”的事实:
interface Billable{public int getRate();public boolean hasSpecialSkill();
}
然后,声明让Employee实现这个接口
class Employee implements Billable
完成以后,修改charge()函数声明,强调改函数只使用Employee的这部分行为
double charge(Billable emp, int days){int base = emp.getRate() * days;if(emp.hasSpecialSkill()){return base * 1.05;}else{return base;}
}
到目前为止,我们只不过是在文档化方面有一点收获。单就这一个函数而言,这样的收获并没有太大价值;但如果有若干个类都是用Billable接口,它就很有用。如果我还想计算电脑租金,巨大的收获就显露出来了:要想计算客户租用电脑的费用,只需让Computer类实现Billable接口,然后就可以把租用电脑的时间也填到时间表上了。