• 详细介绍设计模式七大原则


    1. 概述

    设计模式的七大原则旨在提高软件的可维护性、可复用性和可扩展性,包括:

    1. 单一职责原则:一个类应该只有一个引起它变化的原因。
    2. 开闭原则:软件实体应对扩展开放,对修改封闭。
    3. 里氏替换原则:子类型必须能够替换掉它们的基类型。
    4. 依赖倒置原则:高层模块不应依赖于低层模块,两者都应依赖于抽象;抽象不应依赖于细节,细节应依赖于抽象。
    5. 接口隔离原则:使用多个专门的接口比使用单一的总接口更好。
    6. 合成/聚合复用原则:尽量使用对象的组合/聚合,而不是继承关系达到复用的目的。
    7. 迪米特法则(最少知道原则):一个对象应对其他对象有尽可能少的了解。

    这些原则指导开发者设计出更加健壯、灵活、易于维护的软件系统。

    2. 单一职责原则

    单一职责原则(Single Responsibility Principle, SRP)指一个类应该仅有一个引起它变化的原因。这意味着一个类应该只负责一项职责。
    通俗地讲,单一职责原则就像是说,一个人只应该有一个工作。想象你有一个朋友,他既是厨师也是司机。如果有一天他因为烹饪而分心,导致开车出事了,那就是因为他承担了太多的责任。在编程中,如果一个类同时负责多件事情(比如,既存储数据又显示数据),那么当其中一部分需要改变时,很容易影响到其他部分。遵循单一职责原则,意味着每个类只负责一件事情,这样当需求变化时,只需修改有限的部分,减少错误,使代码更容易维护和理解。

    反面案例(违反SRP)

    class User {
        private String userName;
        private String userEmail;
    
        public void saveUser(User user) {
            // 保存用户信息到数据库
        }
    
        public void printUserDetails() {
            // 打印用户详情
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    这个User类违反了单一职责原则,因为它既处理用户信息的保存,又处理用户信息的打印,这两个职责应该由不同的类来承担。

    正面案例(遵循SRP)

    为了遵守单一职责原则,我们可以将上述User类分解为两个类:一个负责用户数据的管理,另一个负责用户数据的显示。

    class User {
        private String userName;
        private String userEmail;
    
        // 仅包含用户数据和基本操作
    }
    
    class UserPersistence {
        public void saveUser(User user) {
            // 保存用户信息到数据库
        }
    }
    
    class UserDisplay {
        public void printUserDetails(User user) {
            // 打印用户详情
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    通过这种方式,我们将用户信息的保存和显示分别封装到了UserPersistenceUserDisplay类中,每个类都只负责单一的功能,从而遵循了单一职责原则。

    Java生态系统中单一职责原则的体现可以在其标准库中找到许多例子,例如:

    • java.io包:专门用于输入输出操作,其中的类如FileReaderBufferedReader等都专注于单一功能。
    • java.util包:提供了一系列工具类和接口,如ListMapSet接口分别专注于不同类型的集合操作。
    • javax.servlet.http.HttpServlet:在开发Web应用时,这个类允许你通过重写doGetdoPost方法来处理HTTP GET或POST请求,而不必同时处理两者,遵循了单一职责。

    这些类和接口的设计遵循了单一职责原则,每个类或接口都专注于一组特定的功能,使得Java生态系统更加模块化和易于维护。

    3. 开闭原则

    开闭原则(Open-Closed Principle, OCP)是面向对象设计的核心原则之一,它指出软件实体(类、模块、函数等)应该对扩展开放,对修改封闭。这意味着在不修改现有代码的情况下,应该能够添加新功能。

    开闭原则就像是给你的应用程序装上了一个“扩展插槽”,让你可以随时增加新功能而不需要打开机器去重新焊接电路板。想象一下,如果你有一个游戏机,每当出现新游戏时,你不需要更换游戏机内部的硬件就能玩,只需要购买新的游戏卡带插上去即可。这样,游戏机的设计就允许了扩展(新增游戏),而不需要修改(打开游戏机更换部件),这正是开闭原则的精髓。

    反面案例

    假设我们有一个根据用户类型计算折扣的系统:

    class DiscountCalculator {
        public double calculateDiscount(String userType) {
            if ("VIP".equals(userType)) {
                return 0.2;
            } else {
                return 0.1;
            }
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    如果需要添加新的用户类型和折扣规则,我们必须修改DiscountCalculator类,违反了开闭原则。

    正面案例

    为了遵循开闭原则,我们可以定义一个接口和多个实现该接口的类,每个类对应一种用户类型的折扣计算方式:

    interface DiscountStrategy {
        double calculateDiscount();
    }
    
    class VIPDiscount implements DiscountStrategy {
        public double calculateDiscount() {
            return 0.2;
        }
    }
    
    class RegularDiscount implements DiscountStrategy {
        public double calculateDiscount() {
            return 0.1;
        }
    }
    
    class DiscountCalculator {
        public double calculateDiscount(DiscountStrategy strategy) {
            return strategy.calculateDiscount();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    这样,当需要添加新的用户类型时,只需添加新的DiscountStrategy实现类,无需修改现有的DiscountCalculator类,从而遵循了开闭原则。

    4. 里氏替换原则

    里氏替换原则(Liskov Substitution Principle, LSP)是面向对象设计的一个原则,它指出如果类A是类B的一个子类型,那么程序中使用类B的地方就可以不改变程序的行为的前提下,用类A来替换类B。

    里氏替换原则就像是说,如果你有一个苹果,那么任何地方需要苹果的地方,你都可以用一个红苹果或一个绿苹果来替代,而不会影响到使用它的地方的正常运作。在编程中,如果你用一个子类对象替换了一个父类对象,那么程序还应该能够像原来一样运行,而不会出现错误或者异常行为。这就要求子类在继承父类的时候,不仅要继承父类的特性,还要保证这些特性的行为不被改变。

    反面案例

    假设我们有一个矩形类和一个正方形类,正方形继承自矩形:

    class Rectangle {
        protected int width, height;
    
        public void setWidth(int width) {
            this.width = width;
        }
    
        public void setHeight(int height) {
            this.height = height;
        }
    
        public int getArea() {
            return width * height;
        }
    }
    
    class Square extends Rectangle {
        public void setWidth(int width) {
            super.setWidth(width);
            super.setHeight(width);
        }
    
        public void setHeight(int height) {
            super.setWidth(height);
            super.setHeight(height);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

    这里,Square违反了LSP,因为它改变了Rectangle的行为,使得设置宽度或高度的操作同时改变了另一方。

    正面案例

    为了遵循LSP,我们可以抽象出一个共同的基类或接口,然后让RectangleSquare分别实现它们自己的行为:

    interface Shape {
        int getArea();
    }
    
    class Rectangle implements Shape {
        protected int width, height;
    
        public Rectangle(int width, int height) {
            this.width = width;
            this.height = height;
        }
    
        @Override
        public int getArea() {
            return width * height;
        }
    }
    
    class Square implements Shape {
        private int side;
    
        public Square(int side) {
            this.side = side;
        }
    
        @Override
        public int getArea() {
            return side * side;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30

    在这个例子中,RectangleSquare都遵循LSP,因为它们都独立实现了Shape接口,且使用它们的代码不需要知道具体是哪一个形状类,从而保证了类型的替换不会导致程序行为的改变。

    5.依赖倒置原则

    依赖倒置原则(Dependency Inversion Principle, DIP)指的是高层模块不应该依赖于低层模块,两者都应该依赖于抽象;抽象不应该依赖于细节,细节应该依赖于抽象。这个原则的核心在于促进系统的解耦,从而使得系统更易于扩展和维护。

    依赖倒置原则(Dependency Inversion Principle, DIP)的核心思想是高层模块不应该依赖低层模块,它们都应该依赖于抽象;抽象不应该依赖细节,细节应该依赖抽象。用通俗的语言来说,就像是建筑的设计不应该基于具体的砖块类型,而是基于砖块的一般特性。这样,无论使用什么样的砖块,只要符合这些特性,就能构建出建筑。在编程中,这意味着我们的代码应该依赖于接口或抽象类,而不是具体的实现类,这样可以使代码更灵活、更易于维护和扩展。

    反面案例

    假设我们有一个LightBulb类和一个ElectricPowerSwitch类,后者依赖于前者:

    class LightBulb {
        public void turnOn() {
            // 实现开灯
        }
    
        public void turnOff() {
            // 实现关灯
        }
    }
    
    class ElectricPowerSwitch {
        private LightBulb lightBulb;
    
        public ElectricPowerSwitch(LightBulb lightBulb) {
            this.lightBulb = lightBulb;
        }
    
        public void press() {
            // 实现开关灯
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    这个例子违反了DIP,因为ElectricPowerSwitch直接依赖于LightBulb的具体实现。

    正面案例

    为了遵循DIP,我们可以引入一个抽象的接口,使得高层和低层模块都依赖于这个接口:

    interface Switchable {
        void turnOn();
        void turnOff();
    }
    
    class LightBulb implements Switchable {
        public void turnOn() {
            // 实现开灯
        }
    
        public void turnOff() {
            // 实现关灯
        }
    }
    
    class ElectricPowerSwitch {
        private Switchable device;
    
        public ElectricPowerSwitch(Switchable device) {
            this.device = device;
        }
    
        public void press() {
            // 实现控制任何Switchable设备
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26

    在这个例子中,通过引入Switchable接口,ElectricPowerSwitch不再依赖于LightBulb的具体实现,而是依赖于抽象。这样,我们可以轻松地用另一个实现了Switchable接口的类来替换LightBulb,比如风扇,而不需要修改ElectricPowerSwitch类的代码。

    6.接口隔离原则

    接口隔离原则(Interface Segregation Principle, ISP)强调不应该强迫客户依赖于它们不使用的接口。换句话说,更倾向于创建专门的接口而不是一个大而全的接口。

    接口隔离原则(Interface Segregation Principle, ISP)讲的是不应该强迫客户依赖于它们不用的接口。用一个简单的例子来说,如果有一个多功能打印机,它可以打印、扫描和复印。根据接口隔离原则,我们不应该只有一个接口包含所有这些功能,因为不是每个使用打印机的人都需要扫描和复印的功能。相反,应该为打印、扫描和复印各自提供独立的接口。这样,只需要打印功能的用户就不必实现或依赖于扫描和复印的接口了。简而言之,接口隔离原则就是让接口更小、更专注,避免一个庞大的接口承担太多的职责。

    反面案例

    假设我们有一个接口包含了太多的方法,客户类必须实现它们即使不需要所有的方法:

    interface Worker {
        void work();
        void eat();
    }
    
    class HumanWorker implements Worker {
        public void work() {
            // 实现工作
        }
    
        public void eat() {
            // 实现吃饭
        }
    }
    
    class RobotWorker implements Worker {
        public void work() {
            // 实现工作
        }
    
        public void eat() {
            // 机器人不需要吃饭,但依然需要实现该方法
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    正面案例

    遵循接口隔离原则,我们应该将Worker接口拆分为更小的、更专门的接口:

    interface Workable {
        void work();
    }
    
    interface Eatable {
        void eat();
    }
    
    class HumanWorker implements Workable, Eatable {
        public void work() {
            // 实现工作
        }
    
        public void eat() {
            // 实现吃饭
        }
    }
    
    class RobotWorker implements Workable {
        public void work() {
            // 实现工作
        }
        // 机器人不需要实现Eatable接口
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    这样,每个类只需实现它实际需要的接口,避免了不必要的依赖。

    7. 合成/聚合复用原则

    合成/聚合复用原则(Composition/Aggregation Reuse Principle, CARP)强调使用对象的组合或聚合来实现新功能,而不是通过继承关系。这样做可以提高代码的灵活性和复用性。

    合成/聚合复用原则就像是搭积木。想象你正在建造一个小屋,你可以选择用预制的部分(比如窗户、门等)来组合成你想要的结构,而不是自己从头开始制造每一个部分。在编程中,这个原则告诉我们应该通过将现有的对象(积木块)组合起来来创建新的功能,而不是通过继承一个大而全的类(从零开始造一个整体)。这样做使得代码更加灵活,因为你可以随时替换或者重新组合这些“积木块”,而不是被固定在一种设计之中。

    合成/聚合复用原则鼓励使用对象的组合或聚合来实现功能的复用,而不是通过继承。这是因为继承会导致强耦合关系,使得父类和子类之间的依赖关系过于紧密,这样一来,修改父类可能会影响到所有的子类,增加了代码的复杂性和维护难度。相比之下,组合或聚合能够提供更加灵活的复用机制,对象之间的关系更加松散,修改一个对象不会直接影响到使用它的其他对象,这样有助于降低系统的耦合度,提高代码的可维护性和可扩展性。

    反面案例

    通过继承实现复用:

    class Vehicle {
        void startEngine() {
            // 启动引擎的代码
        }
    }
    
    class Car extends Vehicle {
        // Car 类继承了 Vehicle 类,复用了 startEngine 方法
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    这种方式使Car类与Vehicle类紧密耦合,限制了灵活性。

    正面案例

    使用组合来实现复用:

    class Engine {
        void start() {
            // 启动引擎的代码
        }
    }
    
    class Car {
        private Engine engine;
    
        Car(Engine engine) {
            this.engine = engine;
        }
    
        void startEngine() {
            engine.start();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17

    在这个例子中,Car类通过包含Engine类的实例(组合)来复用启动引擎的功能,而不是继承Vehicle类。这样Car类就可以更灵活地复用其他类的功能,同时也降低了类之间的耦合度。

    8. 迪米特法则

    迪米特法则(Law of Demeter, LoD),也称为最少知识原则,是一种软件开发的设计指导原则。它强调,一个对象应该对其他对象有尽可能少的了解,只与直接的朋友通信。直接的朋友指的是成员变量、方法参数或者对象创建的实例。

    迪米特法则就像是说,一个人应该尽可能少地知道其他人的私事,只和直接的朋友交流。在编程中,这意味着一个类不应该知道太多其他类的细节,只和直接相关的类交互。这样做可以减少系统中的耦合,使得修改一个部分的时候,不会影响到太多其他部分,保持代码的整洁和可维护性。

    反面案例

    class Paper {
        public String getContent() {
            return "content";
        }
    }
    
    class Printer {
        public void printPaper(Paper paper) {
            System.out.println(paper.getContent());
        }
    }
    
    class User {
        void print() {
            Printer printer = new Printer();
            Paper paper = new Paper();
            printer.printPaper(paper); // User 类直接与 Paper 类交互
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    在这个例子中,User 类需要知道Paper 类的细节才能打印内容,违反了迪米特法则。

    正面案例

    class Paper {
        private String content = "content";
    
        public void printContent() {
            System.out.println(content);
        }
    }
    
    class Printer {
        public void printPaper(Paper paper) {
            paper.printContent();
        }
    }
    
    class User {
        void print() {
            Printer printer = new Printer();
            Paper paper = new Paper();
            printer.printPaper(paper); // User 类不需要直接了解 Paper 类的内容
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    在这个改进后的例子中,User 类通过Printer 类来打印文档,而不需要直接与Paper 类交互。User 类仅与Printer 类有直接关系,遵循了迪米特法则。

    9. 附加知识

    9.1 聚合、组合、继承的区别

    聚合(Aggregation)表示一种弱“拥有”关系,对象间是整体与部分的关系,但部分可以离开整体而单独存在。

    组合(Composition)表示一种强“拥有”关系,对象间也是整体与部分的关系,但部分不能离开整体而单独存在。

    继承(Inheritance)表示一种“是”关系,用于表示一种类型是另一种类型的特化。

    想象一下:

    • 聚合就像是一支球队和球员的关系。球队由球员组成,但是球员可以离开球队,加入其他球队,球队和球员都可以独立存在。

    • 组合就像是鸟和翅膀的关系。翅膀是鸟的一部分,翅膀不能脱离鸟单独存在。如果鸟不存在了,翅膀也就不存在了。

    • 继承就像是孩子和父母的关系。孩子从父母那里继承了特征(比如眼睛的颜色,头发的类型)。孩子是父母的一个特殊版本,拥有父母的一些特性,同时也可能有自己的一些特性。

    代码示例
    // 聚合示例
    class Engine { }
    class Car {
        private Engine engine; // 聚合关系,Car有一个Engine,但Engine可以独立于Car存在
        Car(Engine engine) {
            this.engine = engine;
        }
    }
    
    // 组合示例
    class Room { }
    class House {
        private List<Room> rooms = new ArrayList<>(); // 组合关系,Room是House的一部分,不能独立存在
    }
    
    // 继承示例
    class Vehicle {
        void start() { }
    }
    class Bicycle extends Vehicle { // 继承关系,Bicycle是Vehicle的一种
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    在这些例子中,CarEngine之间的聚合关系允许Engine脱离Car独立存在;HouseRoom之间的组合关系表明Room不能脱离House独立存在;Bicycle继承自Vehicle,表明每个Bicycle也是一个Vehicle

    9.2 解读:抽象不应依赖于细节,细节应依赖于抽象

    “抽象不应依赖于细节,细节应依赖于抽象”是依赖倒置原则的核心思想。这句话意味着在设计软件时,高层模块(定义应用程序的核心行为)不应该依赖于低层模块(具体实现细节),而是两者都应该依赖于抽象(接口或抽象类)。同样,这些抽象不应该依赖于具体的实现细节,而具体的实现细节应该依赖于抽象。

    遵循“抽象不应依赖于细节,细节应依赖于抽象”的原则具有重要意义,因为它促进了代码的灵活性和可维护性。这样做使得代码对改变更加开放:当系统的具体实现需要变化时,不需要对依赖于抽象的高层模块进行大幅度修改。这降低了代码间的耦合度,使得各个部分更容易理解、测试和重用。简而言之,这个原则帮助开发者构建出易于扩展和修改的系统,减少了未来可能的工作量和复杂性。

    想象一下,你正在建造一栋房子。根据“抽象不应依赖细节,细节应依赖于抽象”的原则,你不应该从门把手或砖块开始规划整栋房子;相反,你应该从房子的设计图纸开始,这些图纸定义了房子的结构和它应该如何建造。

    在编程中,这意味着你的代码(即房子的设计)不应该依赖于具体的实现(即门把手和砖块),而应该依赖于接口或抽象类(即图纸)。这样,如果你决定更换门把手或砖块(即具体的实现),你的代码仍然可以工作,因为它依赖的是抽象,而不是具体的细节。

    // 抽象
    interface Switchable {
        void turnOn();
        void turnOff();
    }
    
    // 细节
    class LightBulb implements Switchable {
        public void turnOn() { /* 实现开灯 */ }
        public void turnOff() { /* 实现关灯 */ }
    }
    
    // 高层模块
    class ElectricPowerSwitch {
        private Switchable device; // 依赖于抽象,而不是具体细节
        public ElectricPowerSwitch(Switchable device) {
            this.device = device;
        }
        public void press() {
            /* 使用 device,而不关心它是怎么实现的 */
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    在这个例子中,无论Switchable背后是LightBulb还是其他任何实现Switchable的设备,ElectricPowerSwitch都可以工作,因为它依赖于抽象而不是具体的细节。

  • 相关阅读:
    Code Llama:Llama 2 学会写代码了!
    移动设备软件开发测试
    基于vue3和element-plus的省市区级联组件
    【开源】基于Vue和SpringBoot的微信小程序的音乐平台
    .NET 序列化枚举为字符串
    Spark SQL数据源 - Hive表
    如何判断一个js对象是否存在循环引用
    Z-DArg-GR-pNA,113711-77-6
    加密市场进入寒冬,是“天灾”还是“人祸”?
    Linux网络-UDP/TCP协议详解
  • 原文地址:https://blog.csdn.net/m0_54187478/article/details/136165351