设计模式---七大原则

简介: 设计模式---七大原则

一. 设计模式的目的


要解决耦合性, 内聚性,可维护性, 可扩展性, 重用性. 使程序程序高内聚低耦合的特性.

 

二. 设计模式的七大原则:



设计模式原则,其实就是程序员在编程时,应当准守的原则,也是各种设计模式的基础

大白话的意思是: 我们要学习23中设计模式, 那么设计模式遵循的是什么原则呢? 就是这七大原则


设计模式常用的七大原则


1. 单一职责原则

2.接口隔离原则

3.依赖倒置原则

4.里氏替换原则

5. 开闭原则

6. 迪米特法则

7. 合成复用原则


三. 设计模式七大原则--单一职责原则


3.1. 单一职责原则


对类来说, 即一个类应该只负责一项职责, 如果类A负责两个不同的职责:职责1, 职责2, 当职责1需求变更而改变A时, 可能造成职责2执行错误, 所以需要将类A的粒度拆解为A1, A2

 

3.2 演变过程讲解


案例1:

先来看一段代码, 很简单的一段代码


package singlePrinciple;
/**
 * 这里将单一职责原则
 * 什么是单一职责原则呢? 单一职责原则指的是一个类只做一件事. 如果一个类A做了2件事: A1和A2
 * 那么当A1发生变化改动代码的时候, 很可能会影响A2, 所以, 我们应该将A1和A2差分开为两个类
 */
public class SignlePrinciple01 {
    public static void main(String[] args) {
        Vehicle vehicle = new Vehicle();
        vehicle.run("摩托车");
        vehicle.run("小汽车");
        vehicle.run("飞机");
    }
}
class Vehicle {
    /**
     * 车...跑...
     */
    public void run(String vehicle) {
        System.out.println(vehicle + "正在马路上跑.....");
    }
}


这运行结果会怎么样呢?


摩托车正在马路上跑.....

小汽车正在马路上跑.....

飞机正在马路上跑.....


我们都看到了问题: 飞机好神奇, 会在马路上跑?

分析1: 这个类Vehicle就不满足单一职责原则. 它包含了马路运行的车, 水里运行的船, 天上飞行的飞机. 这时, 他的职责就不是单一的了.

怎么做呢? 我们应该将其拆解开, 每个类只有一个职责.


案例2:


package singlePrinciple;
/**
 * 这里将单一职责原则
 * 什么是单一职责原则呢? 单一职责原则指的是一个类只做一件事. 如果一个类A做了2件事: A1和A2
 * 那么当A1发生变化改动代码的时候, 很可能会影响A2, 所以, 我们应该将A1和A2差分开为两个类
 */
public class SignlePrinciple02 {
    public static void main(String[] args) {
        VehicleRoad vehicle = new VehicleRoad();
        vehicle.run("摩托车");
        VehicleWater vehicleWater = new VehicleWater();
        vehicleWater.run("轮船");
        VehicleAir vehicleAir = new VehicleAir();
        vehicleAir.run("飞机");
    }
}
class VehicleRoad {
    /**
     * 车...跑...
     */
    public void run(String vehicle) {
        System.out.println(vehicle + "正在马路上跑.....");
    }
}
class VehicleWater {
    /**
     * 车...跑...
     */
    public void run(String vehicle) {
        System.out.println(vehicle + "正在水上运行.....");
    }
}
class VehicleAir {
    /**
     * 车...跑...
     */
    public void run(String vehicle) {
        System.out.println(vehicle + "正在天上飞.....");
    }
}


运行结果:

摩托车正在马路上跑.....

轮船正在水上运行.....

飞机正在天上飞.....

 

分析2: 这里,我们将在马路上跑的, 在水里运行的, 在天上飞的, 各个功能拆分在不同的类中.

   这样每个类就实现了单一职责原则. 每个类只做一件事

优缺点: 优点是符合单一职责原则, 缺点是有些浪费, 每个类只有一个方法, 定义了3各类. 占用了3块内存空间.

   可否优化呢? 看案例3

 

案例3:

package singlePrinciple;
/**
 * 这里将单一职责原则
 * 什么是单一职责原则呢? 单一职责原则指的是一个类只做一件事. 如果一个类A做了2件事: A1和A2
 * 那么当A1发生变化改动代码的时候, 很可能会影响A2, 所以, 我们应该将A1和A2差分开为两个类
 */
public class SignlePrinciple03 {
    public static void main(String[] args) {
        Vehicle2 vehicle = new Vehicle2();
        vehicle.runRoad("摩托车");
        vehicle.runWater("轮船");
        vehicle.runAir("飞机");
    }
}
class Vehicle2 {
    public void runRoad(String vehicle) {
        System.out.println(vehicle + "正在马路上跑.....");
    }
    public void runWater(String vehicle) {
        System.out.println(vehicle + "正在水上运行.....");
    }
    public void runAir(String vehicle) {
        System.out.println(vehicle + "正在天上飞.....");
    }
}

这个运行效果和案例2是一样的.


摩托车正在马路上跑.....

轮船正在水上运行.....

飞机正在天上飞.....

分析3: 这种写法符合单一职责原则么? 很显然, 不符合. 单一职责原则说的是一个类只做一件事. 但这里, 一个类做了很多事.


   这种写法可以理解为方法的单一职责. 保证每个方法只做一件事.

     这样不符合单一职责原则, 这样写合适么? 这样写也是可以的. 因为每一个方法都很简单. 不至于像案例2, 很简单的方法却占用一个类.


下面我们在总结一下单一职责原则


3.2 单一职责原则的注意事项和细节


1. 降低类的复杂度, 一个类只负责一个职责

2. 提高类的可读性,可维护性

3. 降低变更引起的风险

4. 通常情况下,我们应当遵守单一职责原则, 只有逻辑足够简单, 才可以在代码级别违反单一职责原则; 只有类中方法数量足够少, 可以在方法级别保持单一职责原则.


我们上面的案例3,就属于类中的方法足够少, 保持在方法级别的单一职责原则.

 

四. 设计模式七大原则--接口隔离原则



4.1 概念


接口隔离原则(Interface Segregation Principle, ISP),定义为:


1. Clients should not be forced to depend upon interfaces that they don’t use. (客户端不应该依赖它不需要的接口。)
2. The dependency of one class to another one should depend on the smallest possible interface. (类间的依赖关系应该建立在最小的接口上。)

接口隔离原则是对接口的使用进行约束规范的一个原则,它告诉我们要想把接口用好,关键在于隔离。


隔离,指断绝接触、断绝往来。那么我们使用接口时,要隔离什么东西呢?


对于上述定义的第1点,“客户端不应该依赖它不需要的接口”,这里的隔离是指客户端和它不需要的接口隔离,也就是客户端不要使用它不需要的接口


我们着重看一下第2点,“类间的依赖关系应该建立在最小的接口上”,它要求“最小的接口”,也就是该接口中没有多余的方法所以这里的隔离是指和多余的方法隔离。


综上所述,接口隔离原则告诉我们,不要把一大堆方法塞进一个接口里,导致这个接口变得臃肿无比。应该要根据实际需要,让接口中只有用得上的方法,也就是说要细化我们的接口。

 

案例分析

1187916-20200616195210171-1318090020.png


这是一个UML图, interface1一共有4个方法,这里B实现了interface1,D也实现了interface1

 

4.2 好处

现在我们知道,接口隔离原则的要点,就是要细化我们的接口。那么这样做具体有什么好处呢?

主要有四个好处,分别是:

1.避免接口污染;

2.提高灵活性;

3.提供定制服务;

4.实现高内聚。

下面就来详细说一下接口隔离原则的好处。

 

4.2.1 避免接口污染


一个类如果要实现一个接口,那么就要实现这个接口要求的所有方法,如果这个接口里面包含这个类不需要的方法,那么就会造成接口污染,这是不好的设计,会对系统留下隐患。

比如说我们有一个枪的接口,枪有两个基本功能:扣动扳机和放子弹。枪还有一个功能:射杀。其接口如下:


package segregationPrinciple;
/**
 * 枪接口
 */
public interface IGun {
    // 扣动扳机
    void trigger();
    // 放子弹
    void bullet();
    // 射杀
    void shot();
}

然后我们现在需要一个玩具枪的类。玩具枪可以扣动扳机, 也可以放子弹,只是不能射杀。为了图方便,我们直接用IGun这个接口来实现我们的玩具枪:

package segregationPrinciple;
public class ToyGun implements IGun {
    @Override
    public void trigger() {
        System.out.println("扣动扳机");
    }
    @Override
    public void bullet() {
        System.out.println("放子弹");
    }
    @Override
    public void shot() {      // 空实现, 什么也不做
    }
}

玩具枪是不能射杀的,但是由于它实现了IGun这个接口,所以只能空实现它并不需要的shot方法,于是玩具枪这个类就被污染了。这好像也没有什么不妥,但是这是有隐患的,因为玩具枪一旦实现了IGun接口,那么在程序里它就代表一把能射杀的枪。假设在后面突然遇到了一个老虎,唯一保命的方法就是拿枪射杀这个老虎,结果你拿到的是你之前为了图方便做的ToyGun,那么你面临的就是灭顶之灾。


4.2.2  提高灵活性

 

一个类是可以同时实现多个接口的,所以将一个臃肿的接口分割为若干个小接口,通过小接口的不同组合可以满足更多的需求。


举个例子。我们现在需要一个代表美女的接口,美女的标准也很明确:面貌、身材和气质,那么我们的美女接口就出来了:


package segregationPrinciple;
public interface IPrettyGirl {
    void goodLooking(); // 好面貌
    void niceFigure(); // 好身材
    void greatTemperament; // 好气质
}

这并没有什么问题。但是在现实中,一定是要美貌和气质兼备的才算美女吗?非也,其实也有长得不好看,但是气质很好的气质美女的,当然也有没有气质但是长得好看的美女。这样上面的接口就不适用了,因为按照上面的的接口,只有长得好看而且气质好的才算美女。

可以通过细化这个接口解决这个问题。上述的接口可以一分为二:

只有外貌好的美女:

package segregationPrinciple;
public interface IGoodBodyGirl {
    void goodLooking(); // 好面貌
    void niceFigure(); // 好身材
}

只有气质好的美女:


package segregationPrinciple;
public interface IGreatTemperamentGirl {
    void greatTemperament();
}

然后通过这两个接口的不同组合,就能满足外貌美女、气质美女和外貌气质俱佳的美女的不同需求了。所以,细化接口可以让我们的接口更加灵活,满足更多需求。


4.2.3 提供定制服务


什么是定制服务?定制服务就是单独为一个个体提供优良的服务。我们在做系统设计时也需要考虑对系统之间或模块之间的接口提供定制服务。提供定制服务就必然有一个需求:只提供访问者需要的方法。这也是可以通过细化接口实现的。


比如我们开发了一个图书管理系统,其中有一个查询图书的接口


package segregationPrinciple;
public interface IBookSearcher {
    void searchByAuthor();//根据作者搜索
    void searchByTitle();//根据书名搜索
    void searchByCatagory();//根据分类搜索
    void complexSearch();//复杂的搜索
}

我们的图书馆管理系统的访问者有管理人员和公网,其中complexSearch方法非常损耗服务器的性能,它只提供给管理人员使用。其他方法管理人员和公网都可以使用。公网这部分是另一个项目组在开发的,所以当时我们口头上跟公网项目组说明不能在公网上调用complexSearch这个方法。图书馆管理系统上线后,有一天发现系统速度非常慢,在熬了一个通宵排查后,发现是由于公网项目组某个程序员的疏忽,把complexSearch方法公布到了公网中...


显然通过口头的方式说哪一个方法不能调用是不管用的,要想彻底解决这个问题,还是得通过细化接口,为访问者定制专有的接口才行。那么上述的接口可以一分为二:

简单的搜索:

package segregationPrinciple;
public interface ISimpleBookSearcher {
    void searchByAuthor();//根据作者搜索
    void searchByTitle();//根据书名搜索
    void searchByCatagory();//根据分类搜索
}

复杂的搜索

package segregationPrinciple;
public interface IComplexBookSearcher {
    void complexSearch();//复杂的搜索
}

这样我们就可以分别给管理人员和公网定制接口了:


  1. 给管理人员提供ISimpleBookSearcherIComplexBookSearcher两个接口;
  2. 给外网提供ISimpleBookSearcher这个接口。


所谓的定制服务,就是通过细化接口,实现给不同的客户提供不同的接口的目的。定制服务可以有效避免因为给客户提供了多余的方法而造成的风险。

4.2.4 高内聚


什么是高内聚?高内聚就是提高接口、类、模块的处理能力,减少对外的交互。比如说,你告诉你的下属“一个小时之内去月球搬一块石头回来”,然后你就躺在海滩上晒着太阳喝着果汁,一个小时之后你的下属就搬着一块月亮上的石头回来给你了。这种不讲任何条件,不需要你关心任何细节,立即完成任务的行为就是高内聚的表现。


具体到接口中,还是尽量细化你的接口。接口是对外界的承诺,承诺越少对系统的开发越有利,变更的风险也就越少,同时也有利于降低成本


4.3 注意


接口隔离原则单一职责原则非常类似。


单一职责原则要求接口的职责是单一的,而接口隔离原则要求接口尽量细化,它们有异曲同工之妙,都是要让我们的接口功能尽量单一,尽量小。


但是,单一职责原则的着重点是在“职责”,而接口隔离原则只单纯地要求接口最小化。


那么,如果已经满足单一职责原则的接口,在当前的需求下还可以继续细化,那么还需要细化吗?


答案是不要再细化了。在实践中,接口设计的粒度越小,系统就越灵活,这是事实。但是灵活的同时也带来了系统的复杂化,导致开发难度增加。


所以接口并不是越小越好,必须要有一个度。当单一职责原则和接口隔离原则存在矛盾时,以满足单一职责原则为底线


相关文章
|
8月前
|
设计模式 前端开发 Java
设计模式之六大基本原则
设计模式之六大基本原则
60 0
|
12天前
|
设计模式 Java 程序员
【23种设计模式·全精解析 | 概述篇】设计模式概述、UML图、软件设计原则
本系列文章聚焦于面向对象软件设计中的设计模式,旨在帮助开发人员掌握23种经典设计模式及其应用。内容分为三大部分:第一部分介绍设计模式的概念、UML图和软件设计原则;第二部分详细讲解创建型、结构型和行为型模式,并配以代码示例;第三部分通过自定义Spring的IOC功能综合案例,展示如何将常用设计模式应用于实际项目中。通过学习这些内容,读者可以提升编程能力,提高代码的可维护性和复用性。
【23种设计模式·全精解析 | 概述篇】设计模式概述、UML图、软件设计原则
|
5月前
|
设计模式
设计模式七大原则
这篇文章介绍了设计模式中的七大原则,特别强调了单一职责原则,即一个类应该只有一个引起其行为变化的原因,以确保类功能的高内聚和低耦合。
|
5月前
|
设计模式 存储 前端开发
React开发设计模式及原则概念问题之自定义Hooks的作用是什么,自定义Hooks设计时要遵循什么原则呢
React开发设计模式及原则概念问题之自定义Hooks的作用是什么,自定义Hooks设计时要遵循什么原则呢
|
7月前
|
设计模式 供应链
设计模式六大原则之迪米特法则
设计模式六大原则之迪米特法则
|
7月前
|
设计模式
设计模式六大原则之依赖倒置原则
设计模式六大原则之依赖倒置原则
|
4月前
|
设计模式 Java 关系型数据库
设计模式——设计模式简介和七大原则
设计模式的目的和核心原则、单一职责原则、接口隔离原则、依赖倒转原则、里氏替换原则、开闭原则、迪米特法则、合成复用原则
|
5月前
|
设计模式 算法 开发者
设计模式问题之最小知识原则(迪米特法则)对代码设计有何影响,如何解决
设计模式问题之最小知识原则(迪米特法则)对代码设计有何影响,如何解决
|
5月前
|
设计模式 前端开发 JavaScript
React开发设计模式及原则概念问题之什么是HOC(Higher-order component),HOC遵循的设计原则都有哪些
React开发设计模式及原则概念问题之什么是HOC(Higher-order component),HOC遵循的设计原则都有哪些
|
5月前
|
设计模式 前端开发 JavaScript
React开发设计模式及原则概念问题之什么是设计模式,单一职责原则如何理解
React开发设计模式及原则概念问题之什么是设计模式,单一职责原则如何理解