当前位置: 首页 > article >正文

重构类关系-Extract Interface提炼接口八

重构类关系-Extract Interface提炼接口八

1.提炼接口

1.1.使用场景

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

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

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

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

Extract Superclass (336)和Extract Interface (341)之间有些相似之处。Extract Interface (341)只能提炼共通接口,不能提炼共通代码。使用Extract Interface (341)可能造成难闻的“重复”坏味道,幸而你可以运用Extract Class (149)先把共通行为放进一个组件中,然后将工作委托该组件,从而解决这个问题。如果有不少共通行为,Extract Superclass (336)会比较简单,但是每个类只能有一个超类。

如果某个类在不同环境下扮演截然不同的角色,使用接口就是个好主意。你可以针对每个角色以Extract Interface (341)提炼出相应接口。另一种可以用上Extract Interface (341)的情况是:你想要描述一个类的外部依赖接口(outbound interface,即这个类要求服务提供方提供的操作)。如果你打算将来加入其他种类的服务对象,只需要求它们实现这个接口即可。

1.2.如何做

  • 新建一个空接口。
  • 在接口中声明待提炼类的共通操作。
  • 让相关的类实现上述接口。
  • 调整客户端的类型声明,令其使用该接口。

1.3.示例

TimeSheet类表示员工为客户工作的时间表,从中可以计算客户应该支付的费用。为了计算这笔费用,TimeSheet需要知道员工级别,以及该员工是否有特殊技能:

// 通过Employee类对象获取员工信息
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的这部分行为

// Billable接口明确表示 只使用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接口,然后就可以把租用电脑的时间也填到时间表上了。


http://www.kler.cn/a/5603.html

相关文章:

  • IntelliJ IDEA 远程调试
  • 每天五分钟机器学习:核函数
  • 源码分析之Openlayers中GeometryCollection类
  • 关于科研中使用linux服务器的集锦
  • 《Mycat核心技术》第06章:Mycat问题处理总结
  • 【JavaEE】Spring Boot 项目创建
  • 【华为】组播IGMP基本原理(5月份将再次更新,这是以前的笔记 还存在些许问题)
  • 异想天开!没有CPU的操作系统
  • c#第二次作业
  • 【Linux内存管理概述】
  • uni-app--》uni-app的生命周期讲解
  • 反向传播自动求微分【Pytorch】
  • 第06章_索引的数据结构
  • 2010年9月计算机二级JAVA笔试试题及答案
  • c语言实践——通讯录(1)(静态版)
  • 机器学习笔记:层次聚类
  • Leetcode.1641 统计字典序元音字符串的数目
  • 《雪国》像憧憬未曾见过的爱恋,美则美矣
  • TCP和UDP网络编程
  • 接收机中的非线性因素来源与模型
  • fastp软件介绍
  • ChatGPT自我分析
  • 【论文写作】如何表示比较关系, compare to OR compare with?
  • 自定义注解:让代码更加简洁优雅
  • 【SpringSecurity】认证授权框架——SpringSecurity使用方法
  • JavaWeb——过滤器Filter和拦截器Interceptor