设计模式之美: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. TinyMCE实现简单的本地上传

    TinyMCE这个东西很多地方再用,不过我以前一直没用过,最近才接触,因为有一套现成的metro风格的皮肤,仅此而已,不过最终如何调用还是我得来实现.其他的都好说,网上的资料一大把一大把的,唯独这个本 ...

  2. wikioi 1002 旁路

    意甲冠军:这个问题刚开始的问题,有错误的含义,原桥始建于一条直线.无论多么遥远. 思维:dfs寻求答案的第一个问题.然后做最小生成树,双方不能大桥将设置INF即可了.然后假设用到INF的边就加上0即可 ...

  3. 手机新闻网站,手持移动新闻,手机报client,jQuery Mobile手机新闻网站,手机新闻网站demo,新闻阅读器开发

    我们坐在地铁.经常拿出新浪手机查看新闻.腾讯新闻,或者看新闻,等刷微信功能.你有没有想过如何实现这些目标. 移动互联网.更活泼. 由于HTML5未来,jQuery Moblie未来. 今天我用jqm的 ...

  4. 运行时间(Java版本)—转换毫秒到时分秒日期

    第一种方式: import java.util.Calendar; import java.util.TimeZone; public class Test { /** * 将毫秒转换为年月日时分秒 ...

  5. hardware_hp存储映射_方案

    修改虚拟磁盘映射方式 每个刀片独立对应映射存储空间 这样就不会造成数据写入冲突, old new 步奏: 创建过程  lun号码  1-155 之间 第二步奏 最后 指定: 就ok了 2012年12月 ...

  6. 使用Navicat Premium 和PL\SQL Developer连接Oracl

    在64位Win7中使用Navicat Premium 和PL\SQL Developer连接Oracle数据库备忘   最近接手了一个项目,服务器端数据库是oracle 11g 64位.由于主要工作不 ...

  7. Android_开发人员经常使用的颜色

    Android开发中经常要用一些个性化的颜色,然而茫茫的RBG颜色对比表,往往给人眼花缭乱的感觉,更别说从中轻易选出一两种比較惬意的颜色,以下我就总结一下开发中经常使用到的比較绚丽的颜色. 以下是经常 ...

  8. git合并远端分支到本地分支的两种方式

    作者:zhanhailiang 日期:2014-11-21 在使用版本号工具提交改动之前,都须要通过update先将本地代码更新到最新版本号.SVN通过svn update就能够实现,那么git怎样实 ...

  9. Java笔试题集锦

    Java笔试题集锦 1.MVC的各个部分都有那些技术来实现?怎样实现? 答:MVC是Model-View-Controller的简写."Model" 代表的是应用的业务逻辑(通过J ...

  10. ftk学习记录(IME文章)

    [声明:版权全部,欢迎转载.请勿用于商业用途. 联系信箱:feixiaoxing @163.com] 前面说的是全屏设置,还是请大家看一下效果图. watermark/2/text/aHR0cDovL ...