C#设计模式之装饰者
IronMan之装饰者
前言
上一篇的文章我们讲到要给"IronMan"配备"武器",并且还使用了"武器",效果还是不错的,对于多种环境、多种攻击方式的"武器"使用,我们已经掌握了。 有的朋友没有看过上一篇文章,那也没关系,此篇的重点不会涉及到上一篇的内容。
好吧,废话不多说,直接进入正题, 这里简要的介绍下,本人一直在为一家"玩具厂"服务,致力于"IronMan"(钢铁侠)的研究,前面的几个篇幅都是在介绍怎么去合理的生产组成"IronMan"的"部件",以及最近需求的变更,需要"部件"可携带武器,并且能攻击,于是在上一篇中讲到了"武器"的使用,感兴趣的朋友可以去看看。 那我们要如何的去把"武器"和"部件"整合在一起呢?
问题的发现
先来看一下本篇篇幅要用到的一些"部件"和"武器"的结构, 部件的:
public abstract class Component
{
///之间的代码于本篇幅无关
private string strName = string.Empty;
/// <summary>
/// 名称
/// </summary>
public string Name
{
get { return strName; }
set { strName = value; }
}
/// <summary>
/// 自我描述
/// </summary>
public abstract void Self_Described();
}
public class RightHandComponent : Component
{
public RightHandComponent() : this("毅代先锋号一代右部件") { }
public RightHandComponent(string strname)
{
base.Name = strname; }
public override void Self_Described()
{
Console.WriteLine("自描述:我是->" + base.Name);
}
}
public class LeftHandComponent : Component
{
public LeftHandComponent() : this("毅代先锋号一代左部件") { }
public LeftHandComponent(string strname)
{
base.Name = strname;
}
public override void Self_Described()
{
Console.WriteLine("自描述:我是->" + base.Name);
}
}
武器的:
(这里的结构经过简化,跟上个篇幅没有联系,感兴趣的朋友可以按照上个篇幅的"武器"结构来设计)
/// <summary>
/// 武器
/// </summary>
public abstract class Weapon
{
/// <summary>
/// 攻击
/// </summary>
public abstract void Attack();
}
/// <summary>
/// 激光武器
/// </summary>
public class LaserWeapon : Weapon
{
public override void Attack()
{
//LaserAttack
Console.WriteLine("激光武器");
}
}
/// <summary>
/// 导弹武器
/// </summary>
public class MissileWeapon : Weapon
{
public override void Attack()
{
//MissileAttack
Console.WriteLine("导弹武器");
}
}
看到这样的结构,会想说怎么去为"部件"提供额外的功能呢?让各种武器继承自"部件"?N种武器怎么办?那"部件"的结构是多么大啊!!!想一下都觉得可怕。
问题的解决
不过还好,有设计模式的存在,在这种特定的情况下首先想到的就是"装饰者"模式。 旧版的"部件"和"武器"已经满足不了现在的需求了(可以满足,但是这样整合起来显得过于复杂和庞大,对于初学者可能不太容易学习,所以这里这样说)。 我们重新来升级"部件"和"武器"的结构:
/// <summary>
/// 新升级的武器规范
/// </summary>
public interface IWeaponUpgrade
{
/// <summary>
/// 是武器了,当然要具备攻击性了,不然也不叫武器
/// </summary>
void Attack();
}
/// <summary>
/// 新升级的部件 支持携带武器(因为已经支持携带了武器,它自然而然的也归纳与武器一类)
/// 此部件为简化版(便于学习)——。
/// </summary>
public class ComUpgrade:IWeaponUpgrade
{
public void Attack()
{
Console.WriteLine("IronMan的某部件开始输出攻击:");
}
}
所以这里的"部件"也就是"武器"了,因为"部件"是被动武器,它自身并没有攻击能力,它需要安装主动性的武器(也就是需要给它装饰上真正具有攻击性的武器),
那我们再来看一下主动性攻击武器的结构:
/// <summary>
/// 武器包装器,主动性攻击武器要实现。功能:可以塞入任何武器类型,并且使得塞入的武器获得此武器(可是实现了此类的子类)的功能
/// </summary>
public class WeaponDecorator : IWeaponUpgrade
{
protected IWeaponUpgrade weaponUpgrade;
public WeaponDecorator(IWeaponUpgrade weapon)
{
this.weaponUpgrade = weapon;
}
public virtual void Attack()
{
this.weaponUpgrade.Attack();
}
}
/// <summary>
/// 匕首
/// </summary>
public class Knife : WeaponDecorator
{
public Knife(IWeaponUpgrade weapon)
: base(weapon)
{ }
public override void Attack()
{
base.Attack();
Console.WriteLine("匕首攻击");
}
}
/// <summary>
/// 锤子
/// </summary>
public class Hammer : WeaponDecorator
{
public Hammer(IWeaponUpgrade weapon)
: base(weapon)
{ }
public override void Attack()
{
base.Attack();
Console.WriteLine("锤子攻击");
}
}
要使用的结构都定义好了,现在来看一下使用的情况:
ComUpgrade comupgrade = new ComUpgrade();
Knife knife = new Knife(comupgrade);
Hammer hammer = new Hammer(knife);
hammer.Attack();
看一下结果 图1

这样的结构就是比较完美的了,"部件"(也就是被动武器)的变化或者是装饰武器(主动性武器)的变化两者间是不会受影响的,可以随便新增新的"武器"到"部件"上,而且装饰武器之间也是可以相互装饰的,对于扩展"部件"的功能,这种方式比继承漂亮多了。细心的人会发现,还有跟装饰的顺序有关系,是这样的。 对于功能的扩展或许体现不出来装饰顺序的优美,但是在业务流程的需求里就能体现了,这一点就好比是那种业务链,比如是:网上商场的业务,选择商品->加入购物车->付款(这里只是举例),这样的一条业务,也可以是 选择商品->付款,是的,都是可以直接付款的,这里就可以把“加入购物车”、“付款”定义为装饰者对象,可以把用户信息定义为被装饰者,可以是“加入购物车”然后再“付款”,也可是直接“付款”,想想看这样用装饰者模式是不是很简单的就实现了。
END
作者:金源
出处:http://www.cnblogs.com/jin-yuan/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面
C#设计模式之装饰者的更多相关文章
- Java 设计模式泛谈&装饰者模式和单例模式
设计模式(Design Pattern) 1.是一套被反复使用.多人知晓的,经过分类编目 的 代码设计经验总结.使用设计模式是为了可重用代码,让代码更容易维护以及扩展. 2.简单的讲:所谓模式就是得到 ...
- C#设计模式(9)——装饰者模式(Decorator Pattern)
一.引言 在软件开发中,我们经常想要对一类对象添加不同的功能,例如要给手机添加贴膜,手机挂件,手机外壳等,如果此时利用继承来实现的话,就需要定义无数的类,如StickerPhone(贴膜是手机类).A ...
- 设计模式之装饰者模式-java实例
设计模式之装饰者模式 需求场景 我们有了别人提供的产品,但是别人提供的产品对我们来说还不够完善,我们需要对这个产品的功能进行补强,此时可以考虑使用装饰者模式. 我们已经有了产品,而且这个产品的功能非常 ...
- python设计模式之装饰器详解(三)
python的装饰器使用是python语言一个非常重要的部分,装饰器是程序设计模式中装饰模式的具体化,python提供了特殊的语法糖可以非常方便的实现装饰模式. 系列文章 python设计模式之单例模 ...
- 设计模式:装饰器(Decorator)模式
设计模式:装饰器(Decorator)模式 一.前言 装饰器模式也是一种非常重要的模式,在Java以及程序设计中占据着重要的地位.比如Java的数据流处理,我们可能看到数据流经过不同的类的包装和 ...
- 设计模式之装饰(Decorator)模式
设计模式之装饰(Decorator)模式 (一)什么是装饰(Decorator)模式 装饰模式,又称为包装模式,它以对客户端透明的方式扩张对象的功能,是继承关系的替代方案之一. 装饰模式可以在不使用创 ...
- Java设计模式 - - 单例模式 装饰者模式
Java设计模式 单例模式 装饰者模式 作者 : Stanley 罗昊 [转载请注明出处和署名,谢谢!] 静态代理模式:https://www.cnblogs.com/StanleyBlogs/p/1 ...
- python 设计模式之装饰器模式 Decorator Pattern
#写在前面 已经有一个礼拜多没写博客了,因为沉醉在了<妙味>这部小说里,里面讲的是一个厨师苏秒的故事.现实中大部分人不会有她的天分.我喜欢她的性格:总是想着去解决问题,好像从来没有怨天尤人 ...
- Java IO 流 -- 设计模式:装饰设计模式
在java IO 流中我们经常看到这样的写法: ObjectOutputStream oos = new ObjectOutputStream( new BufferedOutputStream(ne ...
- PHP设计模式之装饰器模式(Decorator)
PHP设计模式之装饰器模式(Decorator) 装饰器模式 装饰器模式允许我们给一个类添加新的功能,而不改变其原有的结构.这种类型的类属于结构类,它是作为现有的类的一个包装 装饰器模式的应用场景 当 ...
随机推荐
- MySQL开发规范
字段设计 (1)建议使用UNSIGNED存储非负数值. (2)建议使用INT UNSIGNED存储IPV4. (4)INT类型固定占用4字节存储,例如INT(4)仅代表显示字符宽度为4位,不代表存储长 ...
- Java 之 数据库
1.SQL--结构化查询语言 a.分类:①DDL--数据定义语言 ②DQL--数据查询语言 ③DML--数据操作语言 ④DCL--数据控制语言 b.DDL:包括对数据库的创建.使用.删除,对表的创建. ...
- Beginning Scala study note(4) Functional Programming in Scala
1. Functional programming treats computation as the evaluation of mathematical and avoids state and ...
- js获取URL中的参数
js获取URL中的一些参数的意思 location对象 含有当前URL的信息. 属性 href 整个URL字符串. protocol 含有URL第一部分的字符串,如http: host 包含有URL中 ...
- MIT 6.828 JOS学习笔记11 Exercise 1.8
Exercise 1.8 我们丢弃了一小部分代码---即当我们在printf中指定输出"%o"格式的字符串,即八进制格式的代码.尝试去完成这部分程序. 解答: 在这个练 ...
- Glide实现圆角和圆形图片
实现圆形图片 , 传参第一个为上下文, 第二个为角度 package com.hh.beauter.util; import android.content.Context; import andro ...
- css3实现圆形逐渐减少动画
写这个动画刚开始完全没有思路,后来参考网上的资料发现可以用半圆实现,具体原理如下: 1.,一个div作为背景,三个div做出三个半圆出来,一个用于旋转,一个靠左(用于与背景吻合),一个靠右(用于与背景 ...
- 同一台电脑上多个myeclipse破解的问题
因为项目版本的问题,电脑上不得装了个myeclipe10版本的,但是破解之后,原来电脑上的myeclipse2014却显 示没有激活,好吧,我又去把myeclipse2014重新激活了一遍,但是到了m ...
- three.js立方体
<!DOCTYPE html> <html lang="en"> <head> <title>three.js webgl - ge ...
- Android Handler消息传递机制
在Android系统中,类Handler主要有如下两个作用. 在新启动的线程中发送消息. 在主线程中获取.处理消息. 类Handler在实现上述作用时,首先在新启动的线程中发送消息,然后在主线程中获取 ...