一、几种常用的设计模式

news2025/1/13 10:01:15

设计模式分类

  • 创建者模式:对象实例化的模式,创建型模式用于解耦对象的实例化过程。
    常用:单例模式、工厂方法模式、抽象工厂模式、建造者模式 。
    不常用:原型模式
  • 结构型模式:把类或对象结合在一起形成一个更大的结构。
    常用:代理模式、桥接模式、装饰者模式、适配器模式。
    不常用:组合模式、外观模式、享元模式。
  • 行为型模式:类和对象如何交互、划分责任和算法。
    常用:观察者模式、模板模式、策略模式、责任链模式、迭代器模式、状态模式。
    不常用:访问者模式、备忘录模式、命令模式、解释器模式、中介模式。
    在这里插入图片描述

各分类中模式的关键点

  • 单例模式:某个类只能有一个实例,并且提供一个全局的访问点。
  • 简单工厂:一个工厂类根据传入的参数决定创建出那一种产品类的实例。
  • 工厂方法:定义一个创建对象的接口,让子类决定实例化那个类。
  • 抽象工厂:创建相关或依赖对象的家族,而无须明确指定具体类。
  • 建造者模式:封装一个复杂对象的构建过程,并可以按步骤构造。
  • 原型模式:通过复制现有的实例来创建新的实例。
  • 适配器模式:将一个类的方法接口转换成客户希望的另一个接口。
  • 组合模式:将对象组合成树形结构以表示“部分-整体”的层次结构。
  • 装饰模式:动态的给对象添加新的功能。
  • 代理模式:为其它对象提供一个代理以便控制这个对象的访问。
  • 享元模式:通过共享技术来有效的支持大量细粒度的对象。
  • 外观模式:对外提供一个同一的方法,来访问子系统中的一群接口。
  • 桥接模式:将抽象部分和它的实现部分分离,使它们都可以独立的变化。
  • 模板模式:定义一个算法结构,而将一些步骤延迟到子类实现。
  • 解释器模式:给定一个语言,定义它的文法的一种表示,并定义一个解释器。
  • 策略模式:定义一系列算法,把它们封装起来,并且使它们可以相互替换。
  • 状态模式:允许一个对象在其对象内部状态改变时改变它的行为。
  • 观察者模式:对象间的一对多的依赖关系。
  • 备忘录模式:在不破坏封装的前提下,保持对象的内部状态。
  • 中介者模式:用一个中介对象来封装一系列的对象交互。
  • 命令模式:将命令请求封装为一个对象,使得可以用不同的请求来进行参数化。
  • 访问者模式:在不改变数据结构的前提下,增加作用于一组对象元素的新功能。
  • 责任链模式:将请求的发送者和接受者解耦,使得多个对象都有处理这个请求的机会。
  • 迭代器模式:一种遍历访问聚合对象中各个元素的方法,不暴露该对象的内部结构。

设计模式详解

1、单例模式

单例模式,它的定义就是确保某一个类只有一个实例,并且提供一个全局访问点。

单例模式的特定:
1、只有一个实例。
2、自我实例化。
3、提供全局访问点。

单例模式的优点主要就是节约系统资源,提高系统效率,同时也能够严格控制客户对它的访问。
也许就是因为系统中只有一个实例,这样就导致了单例类的职责过重,违背了“单一职责原则”,同时也没有抽象类,所以扩展起来有一定的困难。

单例模式实现方式
1、懒汉式单例:第一次被引用时,才会将自己实例化。
利用双重锁定解决多线程不安全的问题。

    class Singleton
    {
        private static Singleton instance;
        private static readonly object syncRoot = new object();
        //构造函数必须是私有,防止在类的外部被实例化
        private Singleton()
        {
        }
        public static Singleton GetInstance()
        {
        	//先判断实例是否存在,不存在再加锁处理
            if (instance == null)
            {
            	//再同一时刻加了锁的那部分程序只有一个线程可以进入
                lock (syncRoot)
                {
                	/*
                	如果当instance为null并且同时有两个线程调用GetInstance()方法时,
                	它们将都可以通过第一重instance==null的判断。然后由于lock机制,这两个线程则只有一个进入,
                	另一个在外排队等候,必须要其中的一个进入并出来后,另一个才能进入。而此时如果没有了第二重的
                	instance==null的判断,则第一个线程创建了实例,而第二线程还是可以继续在创建新的实例。
                	*/
                    if (instance == null)
                    {
                        instance = new Singleton();
                    }
                }
            }
            return instance;
        }
    }

2、饿汉式单例:静态初始化的方式,是在自己被加载时就将自己实例化。
缺点:需要提前占用系统资源。

    public sealed class Singleton
    {
        private static readonly Singleton instance = new Singleton();
        private Singleton() { }
        public static Singleton GetInstance()
        {
            return instance;
        }
    }

2、工厂方法模式

工厂方法:定义一个用于创建对象的接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到其子类。

工厂方法模式非常符合“开闭原则”,当需要增加一个新的产品时,我们只需要增加一个具体的产品类和与之对应的具体工厂即可,无须修改原有系统。
同时在工厂方法模式中用户只需要知道生成产品的具体工厂即可,无须关心产品的创建过程。
虽然它很好的符合了“开闭原则”,但是由于每新增一个新产品时就需要增加两个类。这样就会导致系统的复杂度增加。

结构图
在这里插入图片描述

    class Program
    {
        static void Main(string[] args)
        {
            //工厂方法模式
            IFactory factory = new HUAWEIFactory ();
            Phone phone= factory.CreatePhone();

            phone.Bells();
            phone.Call();
            phone.Send();

            Console.Read();
        }
    }
    //手机对象类
    class Phone
    {
        public void Send()
        {
            Console.WriteLine("发短信");
        }

        public void Call()
        {
            Console.WriteLine("打电话");
        }

        public void Bells()
        {
            Console.WriteLine("闹铃");
        }
    }
    //华为(具体产品类)
    class HUAWEI: Phone 
    { }
    
    //小米(具体产品类)
    class XiaoMi: Phone 
    { }
    
     //手机工厂(创建手机对象的接口)
    interface IFactory
    {
        Phone CreatePhone();
    }
    
    //生产华为手机的工厂(具体产品工厂类)
    class HUAWEIFactory : IFactory
    {
        public Phone CreatePhone()
        {
            return new HUAWEI();
        }
    }
    
     //生产小米手机的工厂(具体产品工厂类)
    class XiaoMiFactory : IFactory
    {
        public Phone CreatePhone()
        {
            return new XiaoMi();
        }
    }

3、抽象工厂模式

  • 抽象工厂模式:提供一个创建一系列相关或相互依赖对象的接口,而无须指定它们具体的类。

    他允许客户端使用抽象的接口来创建一组相关的产品,而不需要关心实际产出的具体产品是什么。这样一来,客户就可以从具体的产品中被解耦。它的优点是隔离了具体类的生成,使得客户端不需要知道什么被创建了,而缺点就在于新增新的行为会比较麻烦,因为当添加一个新的产品对象时,需要更改接口及其下所有子类。
    新增一个产品,就会增加抽象产品、具体产品类以及修改抽象工厂、具体工厂类

  • 组成角色

    角色关系作用
    抽象工厂(AbstractFactory)具体工厂的父类抽象工厂接口,它里面包含所有的产品创建的抽象方法
    具体工厂(ConcreteFactory)抽象工厂的子类;被外界调用实现抽象工厂方法,创建产品的实例
    抽象产品(AbstractProduct)具体产品的父类描述具体产品的公共接口
    具体产品(ConcreteProduct)抽象产品的子类;工厂类创建的目标类对抽象产品的具体分类实现
  • 结构图
    在这里插入图片描述

  • 实现代码

    class Program
    {
        static void Main(string[] args)
        {
            AbstractFactory factory1 = new ConcreteFactory1();
            Client c1 = new Client(factory1);
            c1.Run();

            AbstractFactory factory2 = new ConcreteFactory2();
            Client c2 = new Client(factory2);
            c2.Run();
            
            Console.Read();
        }
    }
    
    //抽象工厂类
    abstract class AbstractFactory
    {
        public abstract AbstractProductA CreateProductA();
        public abstract AbstractProductB CreateProductB();
    }   
    
    //具体工厂类1;实现抽象工厂类方法
    class ConcreteFactory1 : AbstractFactory
    {
        public override AbstractProductA CreateProductA()
        {
            return new ProductA1();
        }
        public override AbstractProductB CreateProductB()
        {
            return new ProductB1();
        }
    }
	//具体工厂类2;实现抽象工厂类方法
    class ConcreteFactory2 : AbstractFactory
    {
        public override AbstractProductA CreateProductA()
        {
            return new ProductA2();
        }
        public override AbstractProductB CreateProductB()
        {
            return new ProductB2();
        }
    } 
      
    //抽象产品A
    abstract class AbstractProductA
    {
    }
	//抽象产品B
    abstract class AbstractProductB
    {
        public abstract void Interact(AbstractProductA a);
    }  
    //具体产品A1
    class ProductA1 : AbstractProductA
    {
    }
    
	//具体产品B1
    class ProductB1 : AbstractProductB
    {
        public override void Interact(AbstractProductA a)
        {
            Console.WriteLine(this.GetType().Name +
              " interacts with " + a.GetType().Name);
        }
    }
	//具体产品A2
    class ProductA2 : AbstractProductA
    {
    }
	//具体产品B2
    class ProductB2 : AbstractProductB
    {
        public override void Interact(AbstractProductA a)
        {
            Console.WriteLine(this.GetType().Name +
              " interacts with " + a.GetType().Name);
        }
    } 
    
    class Client
    {
        private AbstractProductA AbstractProductA;
        private AbstractProductB AbstractProductB;

        // Constructor 
        public Client(AbstractFactory factory)
        {
            AbstractProductB = factory.CreateProductB();
            AbstractProductA = factory.CreateProductA();
        }

        public void Run()
        {
            AbstractProductB.Interact(AbstractProductA);
        }
    }       

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/19466.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Tilemap瓦片资源

1、Tilemap Tilemap一般称之为 瓦片地图或者平铺地图,是Unity2017中新增的功能,主要用于快速编辑2D游戏中的场景,通过复用资源的形式提升地图多样性 工作原理就是用一张张的小图排列组合为一张大地图 它和SpriteShape都是用于制作2D游戏的…

CEAC 之《企业信息化管理》1

👨‍💻个人主页:微微的猪食小窝 欢迎 点赞👍 收藏⭐ 留言📝 加关注✅! 本文由 微微的猪食小窝 原创 收录于专栏 【CEAC证书】 1综合布线是智能建筑的信息高速公路。 A、正确 B、错误A2直通线的一根双绞线的两端执行不同…

Java基础实战项目-------网上订餐系统

目录 前言 项目需求 项目环境准备 技能点 实现思路 ​编辑 项目总结 完整代码: 前言 已学完Java基础部分的内容,如下 理解程序的基本概念:程序、变量、数据类型 会使用顺序、选择、循环、跳转语句编写程序 会使用数组以及Arrays的…

[附源码]SSM计算机毕业设计智慧教学平台JAVA

项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM mybatis Maven Vue 等等组成,B/S模式 M…

[附源码]java毕业设计生产型企业员工管理系统

项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM mybatis Maven Vue 等等组成,B/S模式 M…

应急响应-账户排查

用户信息排查 在服务器被入侵之后,攻击者可能会建立相关账户,方便进行远程控制。 主要采用一下几种: 直接建立一个新用户;(有时候为了混淆视听,账户名称和系统常用名相似)激活一个系统中的默认用户,但是这…

ArcGIS计算图斑四至坐标原来这么简单!可不要在走弯路哦

时常我们需要去计算图斑的四至坐标 (四至与四至点不一样哦) 很多朋友会去求个 最小边界几何 在与原始图斑相交得到点来算四至 这种方法有许多问题 是不可以取的,我们今天来介绍一下 一个简单的字段计算就解决这个问题 然后嫌麻烦 我们…

jtag调试ls1012a linux-5.3内核

1、jtag连接 OK1012A-C jtag引脚如下: 如果jlink的VCC对外输出供电,那么需要关闭,VCC对外供电导致jtag连接不上。使用引脚匹配的转接板连接开发板的jtag插座。使用交叉串口线连接开发板。 2、linux-5.3内核编译 -O0编译修改方法与树莓派4b编译修改方法一…

java基于ssm大学生社团管理系统-计算机毕业设计

系统采用了B/S结构,将所有业务模块采用以浏览器交互的模式,选择MySQL作为系统的数据库,开发工具选择My eclipse来进行系统的设计。基本实现了社团管理应有的主要功能模块,本系统有前台与后台两大功能模块,管理员&#…

【图像隐藏】基于小波变换DWT实现数字水印嵌入提取含各类攻击附matlab代码

✅作者简介:热爱科研的Matlab仿真开发者,修心和技术同步精进,matlab项目合作可私信。 🍎个人主页:Matlab科研工作室 🍊个人信条:格物致知。 更多Matlab仿真内容点击👇 智能优化算法 …

OPT(奥普特)荣摘高工锂电“2022年度创新技术奖”

日前,高工锂电年会暨金球奖颁奖典礼在深圳隆重举行,集结了锂电产业链上下游企业高层领袖,围绕行业新技术、数字工厂、极限智造等共议未来发展之道。 作为锂电行业机器视觉核心供应商,OPT(奥普特)受邀出席年…

【Java八股文总结】之面向对象

文章目录Java面向对象基础一、面向对象基础1、什么是封装?2、什么是继承?1、子类访问父类2、子类的访问修饰符3、方法重写3、什么是多态?1、Java语言如何实现多态2、什么时候使用多态?4、什么是接口?5、怎么使用接口&a…

Stream

目录 一 函数式接口 1 特点 2 核心函数式接口 1) Consumer 2) Supplier 3) Function 4) Predicate 5) 扩展:BiFunction 二 Stream 1 stream操作过程 1) 中间操作 2)终端…

[附源码]java毕业设计汽车票售票系统lunwen

项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM mybatis Maven Vue 等等组成,B/S模式 M…

太湖“生态度假式”年会,为您的健康守护,为您的相聚喝彩

最近几年经常听到, 有人说今年最大的目标就是活着, 历经风雨,方知岁月静好的可贵, 这特殊的一年又一年里让大家深觉“健康”的重要, 也让我们更热爱彼此、热爱生活。 倏忽间,2022已至尾声, 又到…

【ASM】字节码操作 工具类与常用类 GeneratorAdapter 介绍

文章目录 1.概述2. GeneratorAdapter2.1 class info2.2 fields2.3 构造方法2.4 方法2.5 特殊方法2.5.1 loadThis2.5.2 getArgIndex2.5.2 box &3. 案例4.总结1.概述 在上一篇文章中:【ASM】字节码操作 工具类与常用类 AdviceAdapter 介绍 打印方法进入 和 方法退出 的参数…

[附源码]SSM计算机毕业设计远程在线教育平台JAVA

项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM mybatis Maven Vue 等等组成,B/S模式 M…

【蓝桥杯物联网赛项学习日志】Day4 关于USART/UART

关键词:USART UART 串口通信 理论基础 USART/UART USART(Universal Synchronous/Asynchronous Receiver/Transmitter) 通用 同步/异步 串行 接收/发送器是一种串行通信接口。USART最多有5个信号,分别是TX,RX,nCTS.nRTS,SCLK TX串行输出信号RX串行输…

SpringBoot整合freemarker模板导出word文件

文章目录1、前言2、需求说明3、编码3.1、导入依赖3.2、接口编写3.3、工具类3.4、ftl文件3.5、测试4、word转pdf5、总结1、前言 在项目中我们有时间需要根据一个word模板文档,批量生成其他的word文档,里面的有些值改变一下而已,那怎么做呢&am…

【Logback+Spring-Aop】实现全面生态化的全链路日志追踪系统服务插件「SpringAOP 整合篇」

承接前文 针对于上一篇【LogbackSpring-Aop】实现全面生态化的全链路日志追踪系统服务插件「Logback-MDC篇」的功能开发指南之后,相信你对于Sl4fj以及Log4j整个生态体系的功能已经有了一定的大致的了解了,接下来我们需要进行介绍关于实现如何将MDC的编程…