Extract Interface (提炼接口)

Summary:若干客户使用类接口中的同一子集,或者两个类的接口有相同部分。将相同的子集提炼到一个独立接口中。                                java

 动机程序员

类之间彼此相互用的方式有若干种。“使用一个类”一般意味用到该类的全部责任区。另外一种状况是,某一组客户只使用类责任区中的一个特定子集。再一种状况则是,这个类须要与全部协助处理某些特定请求的类合做。函数

对于后两种状况,将真正用到的这部分责任分离出来一般颇有意义,由于这样可使系统的用法更清晰,同时也更容易看清系统的责任划分。若是新的类须要支持上述子集,也比较可以看清子集内有些什么东西。spa

在许多面向对象语言中,这种责任划分是经过多继承(multiple inheritance)来实现的。你能够针对每组行为创建一个类,再将它们组合于同一个实现中。Java只提供单继承(single inheritance),但你能够运用接口来昭示并事项上述需求。接口对于java 程序的设计方式有着巨大的影响,就连Smalltalk程序员都认为接口是一大进步。设计

Extract Superclass Extract Interface之间有些类似之处。Extract Interface 只能提炼共通接口,不能提炼共通代码。使用Extract Interface可能形成难闻的“重复”坏味道,幸而你能够运用Extract Class 先把共通行为放进一个组件中,而后将工做委托该组件,从而解决这个问题。若是有很多共通行为,Extract Superclass会比较简单,可是每一个类只能有一个超类。code

若是某个类在不一样环境下扮演大相径庭的角色,使用接口就是个好主意。你能够针对每一个角色以Extract Interface提炼出相应接口。另外一种能够用上Extract Interface的状况是:你想要描述一个类的外部依赖接口(outbound interface,即这个类要求服务提供方提供的操做)。若是你打算未来加入其余种类的服务对象,只须要求它们实现这个接口便可。对象

作法 继承

1.新建一个空接口接口

2.在接口中声明待提炼类的共通操做。ip

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接口,而后就能够把租用电脑的时间也填到时间表上了。

相关文章
相关标签/搜索