Extract Interface (提炼接口)

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

 动机

类之间彼此相互用的方式有若干种。“使用一个类”通常意味用到该类的所有责任区。另一种情况是,某一组客户只使用类责任区中的一个特定子集。再一种情况则是,这个类需要与所有协助处理某些特定请求的类合作。

对于后两种情况,将真正用到的这部分责任分离出来通常很有意义,因为这样可以使系统的用法更清晰,同时也更容易看清系统的责任划分。如果新的类需要支持上述子集,也比较能够看清子集内有些什么东西。

在许多面向对象语言中,这种责任划分是通过多继承(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接口,然后就可以把租用电脑的时间也填到时间表上了。

转载于:https://my.oschina.net/u/134516/blog/205008

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值