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

C# 设计模式之装饰器模式

总目录


前言

装饰器模式的主要作用就是扩展一个类的功能,或给一个类添加多个变化的情况。学习面向对象的都知道,如果想单纯的给某个类增加一些功能,可以直接继承该类生成一个子类就可以。应对一些简单的业务场景继承也就够了,但是面对一些复杂的业务场景,仅靠继承是不够的。那么我们看看装饰器模式是如果以一个更为灵活的方式扩展一个对象的功能的。


1 基础介绍

  1. 动态地给一个对象添加一些额外的职责,就增加功能来说,装饰器模式相比生成子类更为灵活。
  2. 适用于需要扩展一个类的功能,或给一个类添加多个变化的情况。
  3. 在装饰模式中的角色:
    • 抽象构件角色(Component):给出一个抽象接口,以规范准备接收附加责任的对象。
    • 具体构件角色(Concrete Component):定义一个将要接收附加责任的类。
    • 装饰角色(Decorator):持有一个构件(Component)对象的实例,并实现一个与抽象构件接口一致的接口。
    • 具体装饰角色(Concrete Decorator):负责给构件对象添加上附加的责任。

2 使用场景

当需要扩展一个类的功能或给一个类增加附加责任,并且扩展功能可能存在多个且需要可动态组合配置的情况下,使用装饰器模式就是最好的解决办法

注:如果类的扩展比较简单,并且不会多次进行扩展的情况下直接使用类的继承生成子类的方式更为方便快捷。

3 实现方式

看了上面这些描述,没有案例我们是无法理解装饰器模式的精髓。那么我们现在通过一个手机贴膜,装手机壳的案例来理解一下。

1. 传统模式

(1) 第一个需求,张三有一个手机,现在想给贴膜,代码实现如下:

    public class Phone{public virtual void Show() {Console.WriteLine("手机");}}//贴膜手机 继承自 手机类public class MoPhone : Phone{//贴膜的方法public void TieMo(){Console.WriteLine("给手机贴膜了!");}public override void Show(){base.Show();TieMo();}}

客户端调用:

        static void Main(string[] args){Phone moPhone = new MoPhone();moPhone.Show();Console.ReadKey();}

在这里插入图片描述
(2) 现在需求改了,手机不贴膜了,要装手机壳,于是我们改代码:

    //装手机壳的手机 继承自 手机类public class KePhone : Phone{//装手机壳的方法public void ZhuangKe(){Console.WriteLine("给手机装手机壳了!");}public override void Show(){base.Show();ZhuangKe();}}

客户端调用:

        static void Main(string[] args){Phone phone = new KePhone();phone.Show();Console.ReadKey();}

(3) 现在需求又改了,手机贴膜 + 装手机壳,于是我们改代码:

    public class Phone{public virtual void Show() {Console.WriteLine("手机");}}//贴膜手机 继承自 手机类public class MoPhone : Phone{//贴膜的方法public void TieMo(){Console.WriteLine("给手机贴膜了!");}public override void Show(){base.Show();TieMo();}}//现在又改变注意了,既想给手机贴膜也想给手机装手机壳//直接继承已有的贴膜手机类来实现会比较省事public class KeAndMoPhone : MoPhone{//装手机壳的方法public void ZhuangKe(){Console.WriteLine("给手机装手机壳了!");}public override void Show(){base.Show();ZhuangKe();}}

客户端调用:

        static void Main(string[] args){Phone phone = new KeAndMoPhone();phone.Show();Console.ReadKey();}

在这里插入图片描述
上面的实例中,如果单独贴膜或者单独安装保护壳则直接继承手机类即可。
但如果想要即贴膜又要安装保护壳,各自继承手机类的方式就行不通了,只能在贴膜类或者保护壳类的基础上进行扩展。如果还有添加手机挂饰,那就还需要再一层继承关系,这样就会导致 ”子类爆炸“问题,为了解决这个问题就用到了装饰器,下面看看使用装饰器是怎么给手机添加新功能的。

2. 装饰器模式

1 首先定义手机抽象类 和 手机实现类

    public abstract class AbstractPhone{public abstract void Show();}public class XiaoMiPhone : AbstractPhone{public override void Show(){Console.WriteLine("小米手机");}}

2 再定义一个装饰的抽象类

	//装饰抽象类,是装饰模式的核心public abstract class Decorator : AbstractPhone{//保持对手机对象的引用protected AbstractPhone abstractPhone;public Decorator(AbstractPhone phone){abstractPhone = phone;}public override void Show(){//这行代码比较有意思,是实现装饰模式的巧思abstractPhone?.Show();}}

3 定义装饰抽象类的实现:贴膜装饰,装手机壳装饰

	// 贴膜装饰类,主要实现给手机贴膜的扩展功能public class MoPhone : Decorator{public MoPhone(AbstractPhone phone) : base(phone){}public override void Show(){base.Show();TieMo();}//扩展的功能:贴膜public void TieMo(){Console.WriteLine("给手机贴膜了!");}}
	// 手机壳装饰类,主要实现给手机装手机壳的扩展功能public class KePhone : Decorator{public KePhone(AbstractPhone phone) : base(phone){}public override void Show(){base.Show();    ZhuangKe();}//扩展的功能:装手机壳public void ZhuangKe(){Console.WriteLine("给手机装手机壳了!");}}

客户端调用:

  • 只给手机贴膜
        static void Main(string[] args){AbstractPhone phone1 = new XiaoMiPhone();Decorator decorator1 = new MoPhone(phone1);decorator1.Show();Console.ReadKey();}
  • 只给手机装手机壳
        static void Main(string[] args){AbstractPhone phone2 = new XiaoMiPhone();Decorator decorator2 = new KePhone(phone2);decorator2.Show();Console.ReadKey();}
  • 给手机贴膜+装手机壳
        static void Main(string[] args){AbstractPhone phone3 = new XiaoMiPhone();Decorator decorator3 = new MoPhone(phone3);decorator3 = new KePhone(decorator3);decorator3.Show();Console.ReadKey();}

4 需求变更:现在想给手机 贴膜 + 玩偶吊坠

我们只需新增一个 玩偶吊坠 类 继承自 装饰抽象类,然后定义一个玩偶吊坠的装饰方法

	// 玩偶吊坠装饰类,主要实现给手机装玩偶吊坠的扩展功能public class DiaoZhuiPhone : Decorator{public DiaoZhuiPhone(AbstractPhone phone) : base(phone){}public override void Show(){base.Show();ZhuangDiaoZhui();}//扩展功能:给手机装玩偶吊坠public void ZhuangDiaoZhui(){Console.WriteLine("给手机安装玩偶吊坠了!");}}

客户端调用:

        static void Main(string[] args){//给手机贴膜+玩偶吊坠AbstractPhone phone = new XiaoMiPhone();Decorator decorator = new MoPhone(phone);decorator = new DiaoZhuiPhone(decorator);decorator.Show();Console.ReadKey();}

我们发现当我们想要给手机加新的装饰,只需简单的新增对应的装饰类,在装饰类定义一个扩展的装饰方法(新功能)即可。而且还可以对装饰类进行不同组合,这使得我们的代码非常的灵活。

4 优缺点分析

  • 优点:
    • 相较于继承,装饰器模式可以更为灵活的扩展新功能,并且避免了单独使用继承带来的 “多子类衍生问题“。
    • 很好地符合面向对象设计原则中 ”优先使用对象组合而非继承“和”开放-封闭“原则。装饰者模式有很好地可扩展性。
  • 缺点:装饰者模式会导致设计中出现许多小对象,如果过度使用,会让程序变的更复杂。并且更多的对象会是的差错变得困难,特别是这些对象看上去都很像。

结语

希望以上内容可以帮助到大家,如文中有不对之处,还请批评指正。


参考资料:
C#设计模式之八装饰模式(Decorator Pattern)【结构型】
c#中装饰器模式详解
C#设计模式(9)——装饰者模式(Decorator Pattern)

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 在Linux中宏观的看待线程
  • leetcode17. 电话号码的字母组合,dfs深度优先搜索
  • JC/T 2436-2018 木塑家具板材检测
  • Java 中的 ArrayList 和 LinkedList 在性能上有什么不同?
  • Linux安装Java(JKD)
  • 边缘计算×AI:绘制未来实时智能的宏伟蓝图
  • 智能化的Facebook未来:AI如何重塑社交网络的面貌?
  • Docker-数据卷指令
  • 使用ThreadStatic属性提供线程安全的数据访问
  • 算法学习day30
  • 一天一个Arrays小知识——Arrays.asList()
  • Java在无人驾驶方向的就业方向
  • QT百度智能云API鉴权,查询 文心一言 服务调用情况
  • PXE服务器自助部署
  • Adobe ColdFusion反序列化漏洞(cve-2017-3066)
  • 【402天】跃迁之路——程序员高效学习方法论探索系列(实验阶段159-2018.03.14)...
  • 【跃迁之路】【641天】程序员高效学习方法论探索系列(实验阶段398-2018.11.14)...
  • CSS进阶篇--用CSS开启硬件加速来提高网站性能
  • Git 使用集
  • JavaScript服务器推送技术之 WebSocket
  • JS实现简单的MVC模式开发小游戏
  • Python学习之路16-使用API
  • React-redux的原理以及使用
  • Swoft 源码剖析 - 代码自动更新机制
  • Vue实战(四)登录/注册页的实现
  • 服务器之间,相同帐号,实现免密钥登录
  • 基于Javascript, Springboot的管理系统报表查询页面代码设计
  • 记一次和乔布斯合作最难忘的经历
  • 开发基于以太坊智能合约的DApp
  • 利用阿里云 OSS 搭建私有 Docker 仓库
  • 硬币翻转问题,区间操作
  • ​软考-高级-系统架构设计师教程(清华第2版)【第1章-绪论-思维导图】​
  • ### RabbitMQ五种工作模式:
  • #基础#使用Jupyter进行Notebook的转换 .ipynb文件导出为.md文件
  • (6)添加vue-cookie
  • (done) 两个矩阵 “相似” 是什么意思?
  • (二刷)代码随想录第15天|层序遍历 226.翻转二叉树 101.对称二叉树2
  • (附源码)springboot助农电商系统 毕业设计 081919
  • (附源码)基于SSM多源异构数据关联技术构建智能校园-计算机毕设 64366
  • (接上一篇)前端弄一个变量实现点击次数在前端页面实时更新
  • (使用vite搭建vue3项目(vite + vue3 + vue router + pinia + element plus))
  • (一)基于IDEA的JAVA基础12
  • (原創) 如何解决make kernel时『clock skew detected』的warning? (OS) (Linux)
  • (转)linux 命令大全
  • (转)scrum常见工具列表
  • (转)清华学霸演讲稿:永远不要说你已经尽力了
  • .net core 连接数据库,通过数据库生成Modell
  • .Net CoreRabbitMQ消息存储可靠机制
  • .NET Framework 的 bug?try-catch-when 中如果 when 语句抛出异常,程序将彻底崩溃
  • .NET Framework、.NET Core 、 .NET 5、.NET 6和.NET 7 和.NET8 简介及区别
  • @Autowired和@Resource装配
  • [ CTF ] WriteUp- 2022年第三届“网鼎杯”网络安全大赛(朱雀组)
  • [ vulhub漏洞复现篇 ] Apache APISIX 默认密钥漏洞 CVE-2020-13945
  • [.NET 即时通信SignalR] 认识SignalR (一)
  • [12] 使用 CUDA 进行图像处理