设计模式第十五讲:重构 - 改善既有代码的设计(上)

简介: 设计模式第十五讲:重构 - 改善既有代码的设计

一、第一个案例

如果你发现自己需要为程序添加一个特性,而代码结构使你无法很方便地达成目的,那就先重构这个程序。

在重构前,需要先构建好可靠的测试环境,确保安全地重构。

重构需要以微小的步伐修改程序,如果重构过程发生错误,很容易就能发现错误。

案例分析

影片出租店应用程序,需要计算每位顾客的消费金额。

包括三个类: Movie、Rental(租赁) 和 Customer。

  • 一个客户能租赁多部电影

最开始的实现是把所有的计费代码都放在 Customer 类中。可以发现,该代码没有使用 Customer 类中的任何信息,更多的是使用 Rental 类的信息,因此第一个可以重构的点就是把具体计费的代码移到 Rental 类中,然后 Customer 类的 getTotalCharge() 方法只需要调用 Rental 类中的计费方法即可。

class Customer {
    private List<Rental> rentals = new ArrayList<>();
    void addRental(Rental rental) {
        rentals.add(rental);
    }
    double getTotalCharge() {
        double totalCharge = 0.0;
        for (Rental rental : rentals) {
            switch (rental.getMovie().getMovieType()) {
                case Movie.Type1:
                    totalCharge += rental.getDaysRented();
                    break;
                case Movie.Type2:
                    totalCharge += rental.getDaysRented() * 2;
                    break;
                case Movie.Type3:
                    totalCharge += rental.getDaysRented() * 3;
                    break;
            }
        }
        return totalCharge;
    }
}
class Rental {
    private int daysRented;
    private Movie movie;
    Rental(int daysRented, Movie movie) {
        this.daysRented = daysRented;
        this.movie = movie;
    }
    Movie getMovie() {
        return movie;
    }
    int getDaysRented() {
        return daysRented;
    }
}
class Movie {
    static final int Type1 = 0, Type2 = 1, Type3 = 2;
    private int type;
    Movie(int type) {
        this.type = type;
    }
    int getMovieType() {
        return type;
    }
}
public class App {
    public static void main(String[] args) {
        Customer customer = new Customer();
        Rental rental1 = new Rental(1, new Movie(Movie.Type1));
        Rental rental2 = new Rental(2, new Movie(Movie.Type2));
        customer.addRental(rental1);
        customer.addRental(rental2);
        System.out.println(customer.getTotalCharge());
    }
}

使用 switch 的准则是: 只使用 switch 所在类的数据。解释如下: switch 使用的数据通常是一组相关的数据,例如 getTotalCharge() 代码使用了 Movie 的多种类别数据。当这组类别的数据发生改变时,例如增加 Movie 的类别或者修改一种 Movie 类别的计费方法,就需要修改 switch 代码。如果违反了准则,就会有多个地方的 switch 使用了这部分的数据,那么这些 swtich 都需要进行修改,这些代码可能遍布在各个地方,修改工作往往会很难进行。上面的实现违反了这一准则,因此需要重构。

以下是继承 Movie 的多态解决方案,这种方案可以解决上述的 switch 问题,因为每种电影类别的计费方式都被放到了对应 Movie 子类中,当变化发生时,只需要去修改对应子类中的代码即可

有一条设计原则指示应该多用组合少用继承这是因为组合比继承具有更高的灵活性

  • 第3节
  • 例如上面的继承方案,一部电影要改变它的计费方式,就要改变它所属的类,但是对象所属的类在编译时期就确定了,无法在运行过程中动态改变。(运行时多态可以在运行过程中改变一个父类引用指向的子类对象,但是无法改变一个对象所属的类。)

策略模式就是使用组合替代继承的一种解决方案。引入 Price 类,它有多种实现。Movie 组合了一个 Price 对象,并且在运行时可以改变组合的 Price 对象,从而使得它的计费方式发生改变。

  • 第10.3节

重构后整体的类图和时序图如下:

重构后的代码:

class Customer {
    private List<Rental> rentals = new ArrayList<>();
    void addRental(Rental rental) {
        rentals.add(rental);
    }
    double getTotalCharge() {
        double totalCharge = 0.0;
        for (Rental rental : rentals) {
            totalCharge += rental.getCharge();
        }
        return totalCharge;
    }
}
class Rental {
    private int daysRented;
    private Movie movie;
    Rental(int daysRented, Movie movie) {
        this.daysRented = daysRented;
        this.movie = movie;
    }
    double getCharge() {
        return daysRented * movie.getCharge();
    }
}
interface Price {
    double getCharge();
}
class Price1 implements Price {
    @Override
    public double getCharge() {
        return 1;
    }
}
class Price2 implements Price {
    @Override
    public double getCharge() {
        return 2;
    }
}
class Price3 implements Price {
    @Override
    public double getCharge() {
        return 3;
    }
}
class Movie {
    private Price price;
    Movie(Price price) {
        this.price = price;
    }
    double getCharge() {
        return price.getCharge();
    }
}
class App {
    public static void main(String[] args) {
        Customer customer = new Customer();
        Rental rental1 = new Rental(1, new Movie(new Price1()));
        Rental rental2 = new Rental(2, new Movie(new Price2()));
        customer.addRental(rental1);
        customer.addRental(rental2);
        System.out.println(customer.getTotalCharge());
    }
}

二、重构原则

整体内容如下:

要点列表:

2.1、定义

重构是对软件内部结构的一种调整,目的是在不改变软件可观察行为的前提下,提高其可理解性,降低其修改成本。

2.2、为何重构

  • 改进软件设计
  • 使软件更容易理解
  • 帮助找到 Bug
  • 提高编程速度

2.3、三次法则

第一次做某件事时只管去做;第二次做类似事情时可以去做;第三次再做类似的事,就应该重构。

2.4、间接层与重构

计算机科学中的很多问题可以通过增加一个间接层来解决,间接层具有以下价值:

  • 允许逻辑共享
  • 分开解释意图和实现
  • 隔离变化
  • 封装条件逻辑

重构可以理解为在适当的位置插入间接层以及在不需要时移除间接层

2.5、修改接口

如果重构手法改变了已发布的接口,就必须维护新旧两个接口。可以保留旧接口,让旧接口去调用新接口,并且使用 Java 提供的 @Deprecated 将旧接口标记为弃用。

可见修改接口特别麻烦,因此除非真有必要,否则不要发布接口,并且不要过早发布接口。

2.6、何时不该重构

当现有代码过于混乱时,应当重写而不是重构。

一个折中的办法是,将代码封装成一个个组件,然后对各个组件做重写或者重构的决定。

2.7、重构与设计

软件开发无法预先设计,因为开发过程有很多变化发生,在最开始不可能都把所有情况考虑进去。

重构可以简化设计,重构在一个简单的设计上进行修修改改,当变化发生时,以一种灵活的方式去应对变化,进而带来更好的设计。

2.8、重构与性能

为了软代码更容易理解,重构可能会导致性能减低。

在编写代码时,不用对性能过多关注,只有在最后性能优化阶段再考虑性能问题。

应当只关注关键代码的性能,并且只有一小部分的代码是关键代码。

三、代码的坏味道

本章主要介绍一些不好的代码,也就是说这些代码应该被重构。

1、重复代码 Duplicated Code

同一个类的两个函数有相同表达式,则用 Extract Method 提取出重复代码;

两个互为兄弟的子类含有相同的表达式,先使用 Extract Method,然后把提取出来的函数 Pull Up Method 推入父类。

如果只是部分相同,用 Extract Method 分离出相似部分和差异部分,然后使用 Form Template Method 这种模板方法设计模式。

  • 第10.2节

如果两个毫不相关的类出现重复代码,则使用 Extract Class 方法将重复代码提取到一个独立类中。

2、过长函数 Long Method

函数应该尽可能小,因为小函数具有解释能力、共享能力、选择能力。

分解长函数的原则:当需要用注释来说明一段代码时,就需要把这部分代码写入一个独立的函数中。

Extract Method 会把很多参数和临时变量都当做参数,可以用 Replace Temp with Query 消除临时变量,Introduce Parameter Object 和 Preserve Whole Object 可以将过长的参数列变得更简洁。

条件和循环语句往往也需要提取到新的函数中。

3、过大的类 Large Class

应该尽可能让一个类只做一件事,而过大的类做了过多事情,需要使用 Extract Class 或 Extract Subclass。

先确定客户端如何使用该类,然后运用 Extract Interface 为每一种使用方式提取出一个接口。

4、过长的参数列表 Long Parameter List

太长的参数列表往往会造成前后不一致,不易使用。

面向对象程序中,函数所需要的数据通常能在宿主类中找到。

5、发散式变化 Divergent Change

设计原则: 一个类应该只有一个引起改变的原因。也就是说,针对某一外界变化所有相应的修改,都只应该发生在单一类中。

针对某种原因的变化,使用 Extract Class 将它提炼到一个类中。

6、散弹式修改 Shotgun Surgery

一个变化引起多个类修改。

使用 Move Method 和 Move Field 把所有需要修改的代码放到同一个类中。

7、依恋情结 Feature Envy

一个函数对某个类的兴趣高于对自己所处类的兴趣,通常是过多访问其它类的数据,使用 Move Method 将它移到该去的地方,如果对多个类都有 Feature Envy,先用 Extract Method 提取出多个函数。

8、数据泥团 Data Clumps

有些数据经常一起出现,比如两个类具有相同的字段、许多函数有相同的参数,这些绑定在一起出现的数据应该拥有属于它们自己的对象。

使用 Extract Class 将它们放在一起。

9、基本类型偏执 Primitive Obsession

使用类往往比使用基本类型更好,使用 Replace Data Value with Object 将数据值替换为对象。

10、switch 惊悚现身 Switch Statements

具体参见第一章的案例。

11、平行继承体系 Parallel Inheritance Hierarchies

每当为某个类增加一个子类,必须也为另一个类相应增加一个子类。

这种结果会带来一些重复性,消除重复性的一般策略:让一个继承体系的实例引用另一个继承体系的实例。

12、冗余类 Lazy Class

如果一个类没有做足够多的工作,就应该消失。

13、夸夸其谈未来性 Speculative Generality

有些内容是用来处理未来可能发生的变化,但是往往会造成系统难以理解和维护,并且预测未来可能发生的改变很可能和最开始的设想相反。因此,如果不是必要,就不要这么做。

14、令人迷惑的暂时字段 Temporary Field

某个字段仅为某种特定情况而设,这样的代码不易理解,因为通常认为对象在所有时候都需要它的所有字段。把这种字段和特定情况的处理操作使用 Extract Class 提炼到一个独立类中。

15、过度耦合的消息链 Message Chains

一个对象请求另一个对象,然后再向后者请求另一个对象,然后…,这就是消息链。采用这种方式,意味着客户代码将与对象间的关系紧密耦合。

改用函数链,用函数委托另一个对象来处理。

相关文章
|
3月前
|
设计模式 数据库连接 PHP
PHP中的设计模式:提升代码的可维护性与扩展性在软件开发过程中,设计模式是开发者们经常用到的工具之一。它们提供了经过验证的解决方案,可以帮助我们解决常见的软件设计问题。本文将介绍PHP中常用的设计模式,以及如何利用这些模式来提高代码的可维护性和扩展性。我们将从基础的设计模式入手,逐步深入到更复杂的应用场景。通过实际案例分析,读者可以更好地理解如何在PHP开发中应用这些设计模式,从而写出更加高效、灵活和易于维护的代码。
本文探讨了PHP中常用的设计模式及其在实际项目中的应用。内容涵盖设计模式的基本概念、分类和具体使用场景,重点介绍了单例模式、工厂模式和观察者模式等常见模式。通过具体的代码示例,展示了如何在PHP项目中有效利用设计模式来提升代码的可维护性和扩展性。文章还讨论了设计模式的选择原则和注意事项,帮助开发者在不同情境下做出最佳决策。
|
2月前
|
设计模式 算法 数据库连接
PHP中的设计模式:提高代码的可维护性和扩展性
【10月更文挑战第13天】 本文将探讨PHP中常见的设计模式及其在实际项目中的应用。通过对比传统编程方式,我们将展示设计模式如何有效地提高代码的可维护性和扩展性。无论是单例模式确保类的单一实例,还是观察者模式实现对象间的松耦合,每一种设计模式都为开发者提供了解决特定问题的最佳实践。阅读本文后,读者将能更好地理解和应用这些设计模式,从而提升PHP编程的效率和质量。
|
2月前
|
设计模式 SQL 安全
PHP中的设计模式:单例模式的深入探索与实践在PHP开发领域,设计模式是解决常见问题的高效方案集合。它们不是具体的代码,而是一种编码和设计经验的总结。单例模式作为设计模式中的一种,确保了一个类仅有一个实例,并提供一个全局访问点。本文将深入探讨单例模式的基本概念、实现方式及其在PHP中的应用。
单例模式在PHP中的应用广泛,尤其在处理数据库连接、日志记录等场景时,能显著提高资源利用率和执行效率。本文从单例模式的定义出发,详细解释了其在PHP中的不同实现方法,并探讨了使用单例模式的优势与注意事项。通过对示例代码的分析,读者将能够理解如何在PHP项目中有效应用单例模式。
|
3月前
|
设计模式 算法 数据库连接
PHP中的设计模式:提高代码的可维护性与扩展性
设计模式在PHP开发中至关重要,如单例模式确保类仅有一个实例并提供全局访问点,适用于管理数据库连接或日志记录。工厂模式封装对象创建过程,降低系统耦合度;策略模式定义算法系列并使其可互换,便于实现不同算法间的切换。合理选择设计模式需基于需求分析,考虑系统架构,并通过测试驱动开发验证有效性,确保团队协作一致性和代码持续优化。设计模式能显著提升代码质量,解决开发中的设计难题。
34 8
|
3月前
|
设计模式 算法 PHP
PHP中的设计模式:提升代码的灵活性与可维护性
在本文中,我们将深入探讨PHP编程语言中的一种重要概念——设计模式。设计模式是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。它代表了最佳的实践,被有经验的面向对象的软件开发人员所采用。本文将通过具体的实例,展示如何在PHP项目中应用设计模式,以提高代码的灵活性和可维护性。无论你是PHP初学者还是经验丰富的开发者,都能从中获得有价值的见解。
|
3月前
|
设计模式 算法 PHP
PHP中的设计模式:策略模式的深入探索与实践在软件开发的广袤天地中,PHP以其独特的魅力和强大的功能,成为无数开发者手中的得力工具。而在这条充满挑战与机遇的征途上,设计模式犹如一盏明灯,指引着我们穿越代码的迷雾,编写出更加高效、灵活且易于维护的程序。今天,就让我们聚焦于设计模式中的璀璨明珠——策略模式,深入探讨其在PHP中的实现方法及其实际应用价值。
策略模式,这一设计模式的核心在于它为软件设计带来了一种全新的视角和方法。它允许我们在运行时根据不同情况选择最适合的解决方案,从而极大地提高了程序的灵活性和可扩展性。在PHP这门广泛应用的编程语言中,策略模式同样大放异彩,为开发者们提供了丰富的创作空间。本文将从策略模式的基本概念入手,逐步深入到PHP中的实现细节,并通过一个具体的实例来展示其在实际项目中的应用效果。我们还将探讨策略模式的优势以及在实际应用中可能遇到的挑战和解决方案,为PHP开发者提供一份宝贵的参考。
|
3月前
|
设计模式 存储 数据库连接
探索PHP中的设计模式:提高代码的可维护性与扩展性
本文将深入探讨PHP中常用的设计模式,包括单例模式、工厂模式和观察者模式。通过具体的代码示例,展示如何在实际项目中应用这些设计模式,以提高代码的可维护性与扩展性。无论你是PHP初学者还是有一定经验的开发者,都可以通过本文的学习,提升你的编程技巧和项目架构能力。
|
1月前
|
设计模式 安全 Java
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
|
1月前
|
设计模式 开发者 Python
Python编程中的设计模式:工厂方法模式###
本文深入浅出地探讨了Python编程中的一种重要设计模式——工厂方法模式。通过具体案例和代码示例,我们将了解工厂方法模式的定义、应用场景、实现步骤以及其优势与潜在缺点。无论你是Python新手还是有经验的开发者,都能从本文中获得关于如何在实际项目中有效应用工厂方法模式的启发。 ###
|
27天前
|
设计模式 安全 Java
Kotlin - 改良设计模式 - 构建者模式
Kotlin - 改良设计模式 - 构建者模式

热门文章

最新文章

  • 1
    设计模式转型:从传统同步到Python协程异步编程的实践与思考
    60
  • 2
    C++一分钟之-设计模式:工厂模式与抽象工厂
    47
  • 3
    《手把手教你》系列基础篇(九十四)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-下篇(详解教程)
    54
  • 4
    C++一分钟之-C++中的设计模式:单例模式
    65
  • 5
    《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
    43
  • 6
    《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
    70
  • 7
    Java面试题:结合设计模式与并发工具包实现高效缓存;多线程与内存管理优化实践;并发框架与设计模式在复杂系统中的应用
    62
  • 8
    Java面试题:设计模式在并发编程中的创新应用,Java内存管理与多线程工具类的综合应用,Java并发工具包与并发框架的创新应用
    43
  • 9
    Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
    52
  • 10
    Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
    121