Extension Objects(扩展对象)
设计模式之美: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(扩展对象)的更多相关文章
- 设计模式之美:Extension Object(扩展对象)
索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):使用示例结构实现 Extension Object. 实现方式(二):使用泛型实现 IExtensibleObject<T ...
- 大熊君大话NodeJS之------Global Objects全局对象
一,开篇分析 在上个章节中我们学习了NodeJS的基础理论知识,对于这些理论知识来说理解是至关重要的,在后续的章节中,我们会对照着官方文档逐步学习里面的各部分模块,好了该是本文主角登台亮相的时候了,G ...
- Runtime - Associated Objects (关联对象) 的实现原理
主要围绕3个方面说明runtime-Associated Objects (关联对象) 1. 使用场景 2.如何使用 3.底层实现 3.1 实现原理 3.2 关联对象被存储在什么地方,是不是存放在被 ...
- c# ExpandoObject动态扩展对象
js中的Object 对象. php中的stdClass. c# 也有动态可扩展对象 ExpandoObject,需要添加System.Dynamic引用 用法: dynamic model = ne ...
- JS对象-不可扩展对象、密封对象、冻结对象
不可扩展对象 Object.preventExtensions() 仅阻止添加自身的属性.但属性仍然可以添加到对象原型. 可以用 Object.isExtensible(obj) 来判断对象是否可扩展 ...
- javascript 之 扩展对象 jQuery.extend
在JQuery的API手册中,extend方法挂载在JQuery 和 JQuery.fn两个不同的对象上,但在JQuery内部代码实现的是相同的,只是功能各不相同. 官方解释: jQuery.exte ...
- [C#] Extension Method 扩展方法
当我们引用第三方的DLL.或者Visual Studio自己的库的时候,或许会发现这样的一个情况,如果这个类型有一个XX的方法就好了.这时候我们可以用到扩展方法,是我们的代码更加灵活和高效. 这里我举 ...
- js中使用prototype扩展对象方法
//---------------------对象 //1. var HomeContrl = function(){ this.foo = 'bar'; //对象方法 this.intro = fu ...
- GNOME Shell Extension常用扩展
这篇博文的,主要目的是为了方便我和大家安装GNOME扩展.我将我安装过的所有扩展列在此处. 常用扩展 Clipboard Indicator https://extensions.gnome.org/ ...
随机推荐
- REST API出错响应的设计(转)
REST API应用很多,一方面提供公共API的平台越来越多,比如微博.微信等:一方面移动应用盛行,为Web端.Android端.IOS端.PC端,搭建一个统一的后台,以REST API的形式提供服务 ...
- 飞天熊猫游戏源代码android文本
这款游戏是前一段时间完毕的一个项目,飞行熊猫游戏源代码android版.飞行熊猫基于cocos2d游戏引擎开发.包含了谷歌admob广告支持,没有不论什么版权问题.大家能够自由改动和上传应用商店. 1 ...
- Ajax跨域请求数据实例(JSOPN方式)
今天在做取消申请的时候遇到了一个跨域ajax提交的问题. 情景是: 系统A是asp.net的站点,其中包括一个取消申请的接口(get方式通过参数提交到系统的某一个页面,然后返回提交成功或失败) 系统B ...
- [ZZ] python 语言技巧
http://sahandsaba.com/thirty-python-language-features-and-tricks-you-may-not-know.html 感谢原作者 30 Pyt ...
- JavaScript--基于对象的脚本语言学习笔记(三)
事件处理器 1.一个数据校验表单的例程 <html> <head> <title>js练习</title> <meta http-equiv=&q ...
- 蜗牛—JSP学习乘法表页面输出
<%@page import="java.text.SimpleDateFormat" pageEncoding="utf-8"%> <%@ ...
- .net中的设计模式---单例模式,涉及lock的用法
.客户端代码 static void Main(string[] args) { Singleton singleton2 = Singleton.GetInstance(); Singleton s ...
- oracle_删除同一列的重复数据
<!--删除同一列的重复数据 rowid 在orcle中 数据的物理地址---> delete from tbl_over_picture_alarm a where rowid not ...
- 2014年辛星jquery解读第二节
*************jquery的语法****************** 1.jquery是通过选取HTML元素,而且对选取的元素运行某些操作,从而完毕某些特效的. 2.因此,我们在使用jQu ...
- Linux常用命令2--用户问题、文件的打包压缩
Linux常用命令 如何进行用户和群组的创建和更改 [1]groupadd:用于创建新的群组. 语法:groupadd [-option] 用户名:其常用参数有:-g groupadd -g 555 ...