设计模式之美:Extension Objects(扩展对象)

 

索引

意图

预期对象的接口将在未来被扩展。通过额外的接口来定义扩展对象。

Anticipate that an object’s interface needs to be extended in the future.

Additional interfaces are defined by extension objects..

结构

参与者

Subject

  • 定义抽象主体对象。其定义用于查询是否包含特定扩展的接口。

Extension

  • 所有扩展类的抽象基类。可以定义负责管理扩展自身的操作。
  • Extension 知道其是谁的扩展。

ConcreteSubject

  • 具体的主体类。实现基类定义的 GetExtension 方法并返回相应的扩展对象。

AbstractExtension

  • 特定种类的扩展类的抽象基类。

ConcreteExtension

  • 继承并实现 AbstractionExtension 类。实现特定的扩展功能。

适用性

当以下情况成立时可以使用 Extension Objects 模式:

  • 当需要为已有对象额外添加全新的或无法预见的接口时。
  • 抽象的主要接口在不同的客户类中拥有不同的角色时。
  • 无法通过子类型化来扩展接口行为时。

效果

  • 扩展对象促进了增加新扩展功能。
  • 类的主要抽象接口不会过于膨胀。
  • 接口的主要抽象在不同的子系统中可实现不同的角色。
  • 客户类变得更加复杂。
  • 需要控制对扩展对象的滥用。

相关模式

  • Visitor模式可以为层级的类结构增加新的行为。Visitor 模式与 Extension Objects 模式拥有类似的益处。相比 Visitor 模式,Extension Objects 模式不需要一个稳固的类层级结构,并且也不引入循环依赖。
  • Decorator模式是另一个可以扩展类的行为的模式。客户类在使用 Decorator 对象时的透明性比使用 Extension Objects 更好。在使用窄接口或需要增强已知接口时更适合使用 Decorator 模式。
  • Adapter模式支持适配一个已知接口。Extension Objects 模式支持附加的接口。当对象需要对扩展接口进行适配时可以同时使用 Extension Objects 模式和 Adapter 模式。

实现

实现方式(一):使用示例结构实现 Extension Objects。

 1 namespace ExtensionObjectsPattern.Implementation1
2 {
3 public abstract class Subject
4 {
5 public abstract void Operation1();
6 public abstract Extension GetExtension(string extensionType);
7 }
8
9 public abstract class Extension
10 {
11 protected Subject _owner;
12
13 public Extension(Subject owner)
14 {
15 _owner = owner;
16 }
17
18 public abstract void DoSomething();
19 }
20
21 public abstract class AbstractExtension : Extension
22 {
23 public AbstractExtension(Subject owner)
24 : base(owner)
25 {
26 }
27 }
28
29 public class ConcreteExtension : AbstractExtension
30 {
31 public ConcreteExtension(Subject owner)
32 : base(owner)
33 {
34 }
35
36 public override void DoSomething()
37 {
38 // do something
39 _owner.Operation1();
40 }
41 }
42
43 public class ConcreteSubject : Subject
44 {
45 private ConcreteExtension _extension;
46
47 public ConcreteSubject()
48 {
49 _extension = new ConcreteExtension(this);
50 }
51
52 public override void Operation1()
53 {
54 // do something
55 }
56
57 public override Extension GetExtension(string extensionType)
58 {
59 if (extensionType == "some type")
60 {
61 return this._extension;
62 }
63
64 return null;
65 }
66 }
67
68 public class Client
69 {
70 public void TestCase1()
71 {
72 Subject subject = new ConcreteSubject();
73 Extension extension = subject.GetExtension("some type");
74 extension.DoSomething();
75 }
76 }
77 }

实现方式(二):使用泛型实现 IExtensibleObject<T> 接口。

 1 namespace ExtensionObjectsPattern.Implementation2
2 {
3 public abstract class Subject : IExtensibleObject<Subject>
4 {
5 public abstract void Operation1();
6 public abstract IEnumerable<IExtension<Subject>> Extensions { get; }
7 }
8
9 public interface IExtensibleObject<T>
10 where T : class, IExtensibleObject<T>
11 {
12 IEnumerable<IExtension<T>> Extensions { get; }
13 }
14
15 public interface IExtension<T>
16 where T : class, IExtensibleObject<T>
17 {
18 void Attach(T owner);
19 void Detach(T owner);
20 }
21
22 public abstract class Extension : IExtension<Subject>
23 {
24 protected List<Subject> _owners;
25
26 public void Attach(Subject owner)
27 {
28 _owners.Add(owner);
29 }
30
31 public void Detach(Subject owner)
32 {
33 _owners.Remove(owner);
34 }
35
36 public abstract void DoSomething();
37 }
38
39 public class ConcreteExtension : Extension
40 {
41 public override void DoSomething()
42 {
43 // do something
44 foreach (var item in _owners)
45 {
46 item.Operation1();
47 }
48 }
49 }
50
51 public class ConcreteSubject : Subject
52 {
53 private List<Extension> _extensions = new List<Extension>();
54
55 public ConcreteSubject()
56 {
57 Extension extension = new ConcreteExtension();
58 extension.Attach(this);
59
60 _extensions.Add(extension);
61 }
62
63 public override void Operation1()
64 {
65 // do something
66 }
67
68 public override IEnumerable<IExtension<Subject>> Extensions
69 {
70 get
71 {
72 return _extensions;
73 }
74 }
75 }
76
77 public class Client
78 {
79 public void TestCase1()
80 {
81 Subject subject = new ConcreteSubject();
82
83 foreach (var extension in subject.Extensions)
84 {
85 (extension as Extension).DoSomething();
86 }
87 }
88 }
89 }

设计模式之美》为 Dennis Gao 发布于博客园的系列文章,任何未经作者本人同意的人为或爬虫转载均为耍流氓。

 
 
绿色通道: 好文要顶 关注我 收藏该文与我联系 

Extension Objects(扩展对象)的更多相关文章

  1. 设计模式之美:Extension Object(扩展对象)

    索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):使用示例结构实现 Extension Object. 实现方式(二):使用泛型实现 IExtensibleObject<T ...

  2. 大熊君大话NodeJS之------Global Objects全局对象

    一,开篇分析 在上个章节中我们学习了NodeJS的基础理论知识,对于这些理论知识来说理解是至关重要的,在后续的章节中,我们会对照着官方文档逐步学习里面的各部分模块,好了该是本文主角登台亮相的时候了,G ...

  3. Runtime - Associated Objects (关联对象) 的实现原理

    主要围绕3个方面说明runtime-Associated Objects (关联对象) 1. 使用场景 2.如何使用 3.底层实现 3.1  实现原理 3.2 关联对象被存储在什么地方,是不是存放在被 ...

  4. c# ExpandoObject动态扩展对象

    js中的Object 对象. php中的stdClass. c# 也有动态可扩展对象 ExpandoObject,需要添加System.Dynamic引用 用法: dynamic model = ne ...

  5. JS对象-不可扩展对象、密封对象、冻结对象

    不可扩展对象 Object.preventExtensions() 仅阻止添加自身的属性.但属性仍然可以添加到对象原型. 可以用 Object.isExtensible(obj) 来判断对象是否可扩展 ...

  6. javascript 之 扩展对象 jQuery.extend

    在JQuery的API手册中,extend方法挂载在JQuery 和 JQuery.fn两个不同的对象上,但在JQuery内部代码实现的是相同的,只是功能各不相同. 官方解释: jQuery.exte ...

  7. [C#] Extension Method 扩展方法

    当我们引用第三方的DLL.或者Visual Studio自己的库的时候,或许会发现这样的一个情况,如果这个类型有一个XX的方法就好了.这时候我们可以用到扩展方法,是我们的代码更加灵活和高效. 这里我举 ...

  8. js中使用prototype扩展对象方法

    //---------------------对象 //1. var HomeContrl = function(){ this.foo = 'bar'; //对象方法 this.intro = fu ...

  9. GNOME Shell Extension常用扩展

    这篇博文的,主要目的是为了方便我和大家安装GNOME扩展.我将我安装过的所有扩展列在此处. 常用扩展 Clipboard Indicator https://extensions.gnome.org/ ...

随机推荐

  1. Java虚拟机类型卸载和类型更新解析(转)

    转自:http://www.blogjava.net/zhuxing/archive/2008/07/24/217285.html [摘要]          前面系统讨论过java 类型加载(loa ...

  2. DevExpress.XtraReports.UI.XtraReport 动态报表

    原文:DevExpress.XtraReports.UI.XtraReport 动态报表 using System;using System.Collections.Generic;using Sys ...

  3. 【C++模版之旅】项目中一次活用C++模板(traits)的经历

    曾经曾在一个项目中碰到过一个挺简单的问题,但一时又不能用普通常规的方法去非常好的解决,最后通过C++模板的活用,通过traits相对照较巧妙的攻克了这个问题.本文主要想重现问题发生,若干解决方式的比較 ...

  4. java 获取系统变量(环境变量和环境变量)

    前言 环境变量这个概念不陌生, 就是操作系统的环境变量. 系统变量就是java本身维护的变量. 通过 System.getProperty 的方式获取. 对于不同的操作系统来说. 环境变量的处理可能会 ...

  5. 网络资源(6) - EJB视频

    2014_08_24 http://v.youku.com/v_show/id_XMjE0NjE3MDA0.html?f=5227828 01_EJB3.0_下载安装与运行jboss -------- ...

  6. Tyvj P1015 公路骑 (DP)

     叙述性说明 Description 一个特殊的单行道都在每公里公交车站.们乘坐汽车的公里使来付费. 比如例子的第一行就是一个费用的单子. 没有一辆车子行驶超过10公里.一个顾客打算行驶n公里(1 ...

  7. JS动态获取浏览器宽度和高度

    $(window).resize(function() { var width = $(this).width(); var height = $(this).height(); });

  8. 移植X264成功

    http://blog.csdn.net/mashang123456789/article/details/8673426 http://blog.csdn.net/b5w2p0/article/de ...

  9. NYOJ 330 一个简单的数学

    一个简单的数学题 时间限制:3000 ms  |  内存限制:65535 KB 难度:3 描写叙述 zyc近期迷上了数学,一天,dj想出了一道数学题来难住他.算出1/n,但zyc一时答不上来希望大家能 ...

  10. Tomcat剖析(四):Tomcat默认连接器(1)

    Tomcat剖析(四):Tomcat默认连接器(1) 1. Tomcat剖析(一):一个简单的Web服务器 2. Tomcat剖析(二):一个简单的Servlet服务器 3. Tomcat剖析(三): ...