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

java设计模式(持续更新中)

1 设计模式介绍

  1. 设计模式代表了代码的最佳实践,被有经验的开发人员使用。
  2. 设计模式是很多被反复使用并知晓的,主要是对代码和经验的总结。
  3. 使用设计模式是为了重用代码,并让代码更容易被人理解,保证代码的可靠性。
  4. 对接口编程而不是对实现编程
  5. 有限使用对象组合而不是继承关心

2 设计模式七大原则

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

      原文链接:https://blog.csdn.net/m0_54187478/article/details/136165351

     设计模式--七大原则 - 简书 (jianshu.com)

     以下内容均是借鉴上诉链接

3、设计原则详解

3.1单一职责原则

单一职责原则(Single Responsibility Principle, SRP)指一个类应该仅有一个引起它变化的原因。这意味着一个类应该只负责一项职责。即一个类应该只负责一项职责。如类A负责两个不同职责:职责1,职责2。当职责1需求变更而改变A时,可能造成职责2执行错误,所以需要将类A的粒度分解为A1、A2。

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

3.2开闭原则

开闭原则(Open-Closed Principle, OCP)是面向对象设计的核心原则之一,它指出软件实体(类、模块、函数等)应该对扩展开放,对修改封闭。这意味着在不修改现有代码的情况下,应该能够添加新功能。也就是当软件需要变化时,尽量通过扩展软件实体的行为来实现变化,而不是通过修改已有的代码来实现变化。用抽象构建框架,用实现扩展细节。

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

3.3依赖倒置原则

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

比如有个Person类,可以接受Email、QQ和微信的消息。如果都为其提供一个专门的方法,就会让代码非常的冗余:

可以引入一个IReceiver接口,让Person类依赖该接口。这样QQ、微信和Email各自实现IReceiver里面的方法即可:

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

3.4接口隔离原则

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

列如:类A通过接口i依赖B,类C通过接口i依赖类D,如果接口i对于类A和类C来说不是最小接口,那么类B和类D必须去实现他们不需要的方法。

按隔离原则应当这样处理:将接口i拆分为独立的几个接口,将类分别与他们需要的接口建立依赖关系,也就是采用接口隔离原则。

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

3.5迪米特法则

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

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

3.6里氏替换原则

里氏替换原则(Liskov Substitution Principle)要求所有引用基类的地方必须能透明地使用其子类的对象。也就是在继承关系中,子类尽量不要重写父类的方法。继承实际上让两个类耦合性增强了,特别是运行多态比较频繁的时,整个继承体系的复用性会比较差。

比如一种极端情况:一个类继承了另一个类,但却重写了所有方法,那么继承的意义何在?说好的复用呢?        

解决方法是把原来的父类和子类都继承一个更通俗的基类,在适当的情况下,可以通过聚合,组合,依赖等来代替。

 

3.7合成复用原则

合成复用原则(Composite Reuse Principle)就是是尽量使用合成/聚合的方式,而不是使用继承。

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

4、23种设计模式 

4.1创建模式(5种)

4.1.1单例模式

定义:单例模式确保一个类只有一个实例,并提供一个全局访问点。

单例模式可以分为两种:饿汉式懒汉式

优点: 

  1. 全局访问:提供了一个全局访问点,便于控制实例数量。
  2. 资源节约:避免了创建多个对象时的资源浪费。
  3. 线程安全:在多线程环境中,可以保证只创建一个实例。
  4. 控制实例化:可以控制对象的创建过程。

缺点:

  1. 代码耦合:单例模式可能会隐藏一些依赖关系,导致代码耦合。
  2. 可测试性差:由于单例模式是全局的,这使得单元测试变得困难。
  3. 内存浪费:单例对象在程序的整个生命周期内都占用内存。
  4. 滥用:单例模式有时会被滥用,导致程序设计不灵活。

适用场景:

  • 需要确保在整个应用程序中只存在一个实例的情况,如配置管理器、线程池、缓存等。

 方式一:饿汉式单例

/**
 * 预实例化单例类
 * 该类实现了Singleton模式,确保一个类只有一个实例,并提供对该实例的全局访问点
 * 使用静态预实例化方式,避免了同步加载的性能开销,适用于多线程环境
 */
public class PreloadSingleton {
    // 单例实例,静态成员变量,在类加载时完成实例化
    private static PreloadSingleton instance=new PreloadSingleton();

    // 私有构造方法,防止外部通过new创建实例
    private PreloadSingleton(){
        // 初始化操作(如果有的话)
    }

    /**
     * 获取单例实例的方法
     * @return PreloadSingleton的唯一实例
     */
    public static PreloadSingleton getInstance(){
        // 返回单例实例
        return instance;
    }
}

 可以很明显看出这种方式,在没有使用该单例对象时,该对象就被加载到内存,会造成内存的浪费。但是这种方式没有线程安全。

方式二:懒汉式式单例

为了避免内存的浪费,我们可以采用懒加载,即用到该单例对象的时候再创建。

public class Singleton {
    // 单例实例,私有静态变量
    private static Singleton instance ;
    
    /**
     * 私有构造方法,防止外部实例化
     */
    private Singleton(){
        
    }
    /**
     * 获取单例实例的方法
     * 如果实例不存在,则创建一个新的实例并返回
     * 如果实例已经存在,则直接返回该实例
     * @return Singleton的唯一实例
     */
    public static Singleton getInstance(){
        if(instance == null){
            instance = new Singleton();
        }
        return instance;
    }
}

懒汉式虽然不浪费内存,但是无法保证线程的安全。首先,if判断以及其内存执行代码是非原子性的。 当多线程执行时,可能会同时又多个线程,判断if(instance == null)是否处理,可能多个线程判断成立,那可能就会创建多个实例。

那如何保证线程安全呢?

        相信很多人第一时间都会想到加synchronized关键字,synchronized加载getInstace()函数上确实保证了线程的安全。但是,如果要经常的调用getInstance()方法,不管有没有初始化实例,都会唤醒和阻塞线程。为了避免线程的上下文切换消耗大量时间,如果对象已经实例化了,我们没有必要再使用synchronized加锁,直接返回对象。

public class Singleton {
       private static Singleton instance = null;
       private Singleton() {
       };
       public static synchronized Singleton getInstance() {
              if (instance == null) {
                     instance = new Singleton();
              }
              return instance;
       }
}

 这种方式可能会使性能收到影响,因为每次获取实例要加锁。解决思路:我们能不能在第一次创建的时间加锁,后续都不加锁呢。

public class Singleton {
    private static Singleton instance ;

   
    private Singleton(){

    }
    public static Singleton getInstance(){
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

 上述方式也称“饿汉式双检锁”。这样就可以保证线程安全,也能保证性能。


 


 


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

相关文章:

  • 英伟达与甲骨文携手加速企业级AI和数据处理
  • 搭建一个本地 SMTP 服务器
  • [苍穹外卖]-05Redis快速入门
  • 【算法】冒泡排序
  • CSP-J 算法基础 二分查找与二分答案
  • 【Python爬虫】利用爬虫抓取双色球开奖号码,获取完整数据并通过随机森林和多层感知两种模型进行简单的预测
  • VulnHub DC-1-DC-7靶机WP
  • 计算机网络期末试题及答案
  • Java学习路线
  • Redis及其他缓存
  • 数字孪生之-3D可视化
  • Linux系统安装
  • C++20那些事之何时使用可能性属性?
  • 银行业金融机构反洗钱现场检查数据接口规范(试行)
  • 如何升级用 Helm 安装的极狐GitLab Runner?
  • C#发送正文带图片带附件的邮件
  • Eclipse折叠if、else、try catch的{}
  • Git 提取和拉取的区别在哪
  • 【Jupyter Notebook】安装与使用
  • DBeaver 连接 mysql 报错:Public Key Retrieval is not allowed