C#中的设计模式及原理docWord文件下载.docx
- 文档编号:22087884
- 上传时间:2023-02-02
- 格式:DOCX
- 页数:24
- 大小:39.95KB
C#中的设计模式及原理docWord文件下载.docx
《C#中的设计模式及原理docWord文件下载.docx》由会员分享,可在线阅读,更多相关《C#中的设计模式及原理docWord文件下载.docx(24页珍藏版)》请在冰豆网上搜索。
29.Command
30.Interpreter
31.Iterator
32.Mediator
33.Memento
34.Observer
35.State
36.Strategy
37.TemplateMethod
38.Visitor
39.成功采用设计模式的步骤综述
40.强大的通信和培训
41.设计模式采用指导
42.重用实现,不只是设计模式
43.架构中的设计模式
设计模式的形象比喻
44.图书名称图书信息
45.专家推荐
46.内容简介
47.作者简介
Go
设计模式四人组
F(“四人帮”,又称GangofFour,即ErichGamma,RichardHelm,RalphJohnson&
JohnVlissides四人)的《设计模式》,原名《DesignPatterns:
ElementsofReusalbeObject-OrientedSoftware》(1995年出版,出版社:
AddisonWeslyLongman.Inc),第一次将设计模式提升到理论高度,并将之规范化。
该书提出了23种基本设计模式。
时至今日,在可复用面向对象软件的发展过程中,新的设计模式仍然不断出现。
可复用面向对象软件系统现在一般划分为两大类:
应用程序工具箱和框架(Framework),我们平时开发的具体软件都是应用程序,Java的API属于工具箱;
而框架是构成一类特定软件可复用设计的一组相互协作的类,EJB(EnterpriseJavaBeans)是Java应用于企业计算的框架。
框架通常定义了应用体系的整体结构类和对象的关系等等设计参数,以便于具体应用实现者能集中精力于应用本身的特定细节。
框架主要记录软件应用中共同的设计决策,框架强调设计复用,因此框架设计中必然要使用设计模式。
另外,设计模式有助于对框架结构的理解,成熟的框架通常使用了多种设计模式,如果你熟悉这些设计模式,毫无疑问,你将迅速掌握框架的结构,我们一般开发者如果突然接触EJBJ2EE等框架,会觉得特别难学,难掌握,那么转而先掌握设计模式,无疑是给了你剖析EJB或J2EE系统的一把利器。
设计模式的原则
综述
近年来,大家都开始注意设计模式。
那么,到底我们为什么要用设计模式呢?
这么多设计模式为什么要这么设计呢?
说实话,以前我还真没搞清楚。
就是看大家一口一个"
Designpattern"
,心就有点发虚。
于是就买了本"
四人帮"
的设计模式,结果看得似懂非懂:
看的时候好像是懂了,过一会就忘了。
可能是本人比较"
愚钝"
吧:
))最近,有了点感悟。
"
独乐不如众乐"
,与大家分享一下,还望指教!
为什么要提倡"
DesignPattern"
呢?
根本原因是为了代码复用,增加可维护性。
那么怎么才能实现代码复用呢?
OO界有前辈的几个原则:
原则(OpenClosedPrincipal)、里氏代换原则、合成复用原则。
设计模式就是实现了这些原则,从而达到了代码复用、增加可维护性的目的。
此原则是由"
BertrandMeyer"
提出的。
原文是:
Softwareentitiesshouldbeopenforextension,butclosedformodification"
。
就是说模块应对扩展开放,而对修改关闭。
模块应尽量在不修改原(是"
原"
,指原来的代码)代码的情况下进行扩展。
那么怎么扩展呢?
我们看工厂模式"
factorypattern"
:
假设中关村有一个卖盗版盘和毛片的小子,我们给他设计一"
光盘销售管理软件"
我们应该先设计一"
光盘"
接口。
如图:
[pre]______________ |<
>
| |光盘| |_____________| |+卖()| || |_____________|[/pre] 而盗版盘和毛片是其子类。
小子通过"
DiscFactory"
来管理这些光盘。
代码为:
publicclassDiscFactory{ publicstatic光盘getDisc(java/lang/String.java.html"
target="
_blank"
Stringname){ return(光盘)java/lang/Class.java.html"
Class.forName(name).getInstance();
} } 有人要买盗版盘,怎么实现呢?
publicclass小子{ publicstaticvoidmain(java/lang/String.java.html"
String[]args){ 光盘d=DiscFactory.getDisc("
盗版盘"
);
光盘.卖();
} } 如果有一天,这小子良心发现了,开始卖正版软件。
没关系,我们只要再创建一个"
的子类"
正版软件"
就可以了。
不需要修改原结构和代码。
怎么样?
对扩展开放,对修改关闭。
开-闭原则"
工厂模式是对具体产品进行扩展,有的项目可能需要更多的扩展性,要对这个"
工厂"
也进行扩展,那就成了"
抽象工厂模式"
里氏代换原则
里氏代换原则是由"
BarbaraLiskov"
如果调用的是父类的话,那么换成子类也完全可以运行。
比如:
光盘d=new盗版盘();
d.卖();
现在要将"
类改为"
毛片"
类,没问题,完全可以运行。
Java编译程序会检查程序是否符合里氏代换原则。
还记得java继承的一个原则吗?
子类override方法的访问权限不能小于父类对应方法的访问权限。
比如"
中的方法"
卖"
访问权限是"
public"
,那么"
和"
中的"
方法就不能是protected或private,编译不能通过。
为什么要这样呢?
你想啊:
如果"
的"
方法是private。
那么下面这段代码就不能执行了:
可以说:
里氏代换原则是继承复用的一个基础。
合成复用原则
就是说要少用继承,多用合成关系来实现。
我曾经这样写过程序:
有几个类要与数据库打交道,就写了一个数据库操作的类,然后别的跟数据库打交道的类都继承这个。
结果后来,我修改了数据库操作类的一个方法,各个类都需要改动。
牵一发而动全身"
!
面向对象是要把波动限制在尽量小的范围。
在Java中,应尽量针对Interface编程,而非实现类。
这样,更换子类不会影响调用它方法的代码。
要让各个类尽可能少的跟别人联系,"
不要与陌生人说话"
这样,城门失火,才不至于殃及池鱼。
扩展性和维护性才能提高 理解了这些原则,再看设计模式,只是在具体问题上怎么实现这些原则而已。
张无忌学太极拳,忘记了所有招式,打倒了"
玄冥二老"
,所谓"
心中无招"
设计模式可谓招数,如果先学通了各种模式,又忘掉了所有模式而随心所欲,可谓OO之最高境界。
呵呵,搞笑,搞笑!
(JR)
依赖倒转原则
抽象不应该依赖于细节,细节应当依赖于抽象。
要针对接口编程,而不是针对实现编程。
传递参数,或者在组合聚合关系中,尽量引用层次高的类。
主要是在构造对象时可以动态的创建各种具体对象,当然如果一些具体类比较稳定,就不必在弄一个抽象类做它的父类,这样有画蛇添足的感觉
接口隔离原则
定制服务的例子,每一个接口应该是一种角色,不多不少,不干不该干的事,该干的事都要干
抽象类
抽象类不会有实例,一般作为父类为子类继承,一般包含这个系的共同属性和方法。
注意:
好的继承关系中,只有叶节点是具体类,其他节点应该都是抽象类,也就是说具体类 是不被继承的。
将尽可能多的共同代码放到抽象类中。
迪米特法则
最少知识原则。
不要和陌生人说话。
一个模式的四个基本要素
设计模式使人们可以更加简单方便地复用成功的设计和体系结构。
将已证实的技术表述成设计模式也会使新系统开发者更加容易理解其设计思路。
模式名称(patternname)
一个助记名,它用一两个词来描述模式的问题、解决方案和效果。
命名一个新的模式增加了我们的设计词汇。
设计模式允许我们在较高的抽象层次上进行设计。
基于一个模式词汇表,我们自己以及同事之间就可以讨论模式并在编写文档时使用它们。
模式名可以帮助我们思考,便于我们与其他人交流设计思想及设计结果。
找到恰当的模式名也是我们设计模式编目工作的难点之一。
问题(problem)
描述了应该在何时使用模式。
它解释了设计问题和问题存在的前因后果,它可能描述了特定的设计问题,如怎样用对象表示算法等。
也可能描述了导致不灵活设计的类或对象结构。
有时候,问题部分会包括使用模式必须满足的一系列先决条件。
解决方案(solution)
描述了设计的组成成分,它们之间的相互关系及各自的职责和协作方式。
因为模式就像一个模板,可应用于多种不同场合,所以解决方案并不描述一个特定而具体的设计或实现,而是提供设计问题的抽象描述和怎样用一个具有一般意义的元素组合(类或对象组合)来解决这个问题。
效果(consequences)
描述了模式应用的效果及使用模式应权衡的问题。
尽管我们描述设计决策时,并不总提到模式效果,但它们对于评价设计选择和理解使用模式的代价及好处具有重要意义。
软件效果大多关注对时间和空间的衡量,它们也表述了语言和实现问题。
因为复用是面向对象设计的要素之一,所以模式效果包括它对系统的灵活性、扩充性或可移植性的影响,显式地列出这些效果对理解和评价这些模式很有帮助。
一些基本的设计模式
AbstractFactory:
提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。
Adapter:
将一个类的接口转换成客户希望的另外一个接口。
Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。
Bridge:
将抽象部分与它的实现部分分离,使它们都可以独立地变化。
Builder:
将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。
ChainofResponsibility:
为解除请求的发送者和接收者之间耦合,而使多个对象都有机会处理这个请求。
将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它。
Command:
将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化;
对请求排队或记录请求日志,以及支持可取消的操作。
Composite:
将对象组合成树形结构以表示“部分-整体”的层次结构。
它使得客户对单个对象和复合对象的使用具有一致性。
Decorator:
动态地给一个对象添加一些额外的职责。
就扩展功能而言,它比生成子类方式更为灵活。
Facade:
为子系统中的一组接口提供一个一致的界面,Facade模式定义了一个高层接口,这个接口使得这一子系统更加容易使用。
FactoryMethod:
定义一个用于创建对象的接口,让子类决定将哪一个类实例化。
FactoryMethod使一个类的实例化延迟到其子类。
Flyweight:
运用共享技术有效地支持大量细粒度的对象。
Interpreter:
给定一个语言,定义它的文法的一种表示,并定义一个解释器,该解释器使用该表示来解释语言中的句子。
Iterator:
提供一种方法顺序访问一个聚合对象中各个元素,而又不需暴露该对象的内部表示。
Mediator:
用一个中介对象来封装一系列的对象交互。
中介者使各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互。
Memento:
在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。
这样以后就可将该对象恢复到保存的状态。
Observer:
定义对象间的一种一对多的依赖关系,以便当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并自动刷新。
Prototype:
用原型实例指定创建对象的种类,并且通过拷贝这个原型来创建新的对象。
Proxy:
为其他对象提供一个代理以控制对这个对象的访问。
Singleton:
保证一个类仅有一个实例,并提供一个访问它的全局访问点。
State:
允许一个对象在其内部状态改变时改变它的行为。
对象看起来似乎修改了它所属的类。
Strategy:
定义一系列的算法,把它们一个个封装起来,并且使它们可相互替换。
本模式使得算法的变化可独立于使用它的客户。
TemplateMethod:
定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。
TemplateMethod使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。
Visitor:
表示一个作用于某对象结构中的各元素的操作。
它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。
详细描述
从下一节开始,详细描述以下每一种设计模式。
创建型结构型行为型 类FactoryMethodAdapter_ClassInterpreter TemplateMethod 对象AbstractFactory Builder Prototype SingletonAdapter_Object Bridge Composite Decorator Facade Flyweight ProxyChainofResponsibility Command Iterator Mediator Memento Observer State Strategy Visitor 概览
FactoryMethod
结构 意图定义一个用于创建对象的接口,让子类决定实例化哪一个类。
FactoryMethod使一个类的实例化延迟到其子类。
适用性当一个类不知道它所必须创建的对象的类的时候。
当一个类希望由它的子类来指定它所创建的对象的时候。
当类将创建对象的职责委托给多个帮助子类中的某一个,并且你希望将哪一个帮助子类是代理者这一信息局部化的时候。
AbstractFactory
结构 意图提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。
适用性一个系统要独立于它的产品的创建、组合和表示时。
一个系统要由多个产品系列中的一个来配置时。
当你要强调一系列相关的产品对象的设计以便进行联合使用时。
当你提供一个产品类库,而只想显示它们的接口而不是实现时。
Builder
结构 意图将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。
适用性当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式时。
当构造过程必须允许被构造的对象有不同的表示时。
Prototype
结构 意图用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。
适用性当要实例化的类是在运行时刻指定时,例如,通过动态装载;
或者 为了避免创建一个与产品类层次平行的工厂类层次时;
或者 当一个类的实例只能有几个不同状态组合中的一种时。
建立相应数目的原型并克隆它们可能比每次用合适的状态手工实例化该类更方便一些。
Singleton
结构 意图保证一个类仅有一个实例,并提供一个访问它的全局访问点。
适用性当类只能有一个实例而且客户可以从一个众所周知的访问点访问它时。
当这个唯一实例应该是通过子类化可扩展的,并且客户应该无需更改代码就能使用一个扩展的实例时。
Adapter
结构 意图将一个类的接口转换成客户希望的另外一个接口。
Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。
适用性你想使用一个已经存在的类,而它的接口不符合你的需求。
你想创建一个可以复用的类,该类可以与其他不相关的类或不可预见的类(即那些接口可能不一定兼容的类)协同工作。
(仅适用于对象Adapter)你想使用一些已经存在的子类,但是不可能对每一个都进行子类化以匹配它们的接口。
对象适配器可以适配它的父类接口。
Bridge
结构 意图将抽象部分与它的实现部分分离,使它们都可以独立地变化。
适用性你不希望在抽象和它的实现部分之间有一个固定的绑定关系。
例如这种情况可能是因为,在程序运行时刻实现部分应可以被选择或者切换。
类的抽象以及它的实现都应该可以通过生成子类的方法加以扩充。
这时Bridge模式使你可以对不同的抽象接口和实现部分进行组合,并分别对它们进行扩充。
对一个抽象的实现部分的修改应对客户不产生影响,即客户的代码不必重新编译。
(C++)你想对客户完全隐藏抽象的实现部分。
在C++中,类的表示在类接口中是可见的。
有许多类要生成。
这样一种类层次结构说明你必须将一个对象分解成两个部分。
Rumbaugh称这种类层次结构为“嵌套的普化”(nestedgeneralizations)。
你想在多个对象间共享实现(可能使用引用计数),但同时要求客户并不知道这一点。
一个简单的例子便是Coplien的String类[Cop92],在这个类中多个对象可以共享同一个字符串表示(StringRep)。
Composite
结构 意图将对象组合成树形结构以表示“部分-整体”的层次结构。
Composite使得用户对单个对象和组合对象的使用具有一致性。
适用性你想表示对象的部分-整体层次结构。
你希望用户忽略组合对象与单个对象的不同,用户将统一地使用组合结构中的所有对象。
Decorator
结构 意图动态地给一个对象添加一些额外的职责。
就增加功能来说,Decorator模式相比生成子类更为灵活。
适用性在不影响其他对象的情况下,以动态、透明的方式给单个对象添加职责。
处理那些可以撤消的职责。
当不能采用生成子类的方法进行扩充时。
一种情况是,可能有大量独立的扩展,为支持每一种组合将产生大量的子类,使得子类数目呈爆炸性增长。
另一种情况可能是因为类定义被隐藏,或类定义不能用于生成子类。
Facade
结构 意图为子系统中的一组接口提供一个一致的界面,Facade模式定义了一个高层接口,这个接口使得这一子系统更加容易使用。
适用性当你要为一个复杂子系统提供一个简单接口时。
子系统往往因为不断演化而变得越来越复杂。
大多数模式使用时都会产生更多更小的类。
这使得子系统更具可重用性,也更容易对子系统进行定制,但这也给那些不需要定制子系统的用户带来一些使用上的困难。
Facade可以提供一个简单的缺省视图,这一视图对大多数用户来说已经足够,而那些需要更多的可定制性的用户可以越过facade层。
客户程序与抽象类的实现部分之间存在着很大的依赖性。
引入facade将这个子系统与客户以及其他的子系统分离,可以提高子系统的独立性和可移植性。
当你需要构建一个层次结构的子系统时,使用facade模式定义子系统中每层的入口点。
如果子系统之间是相互依赖的,你可以让它们仅通过facade进行通讯,从而简化了它们之间的依赖关系。
Flyweight
结构 意图运用共享技术有效地支持大量细粒度的对象。
适用性一个应用程序使用了大量的对象。
完全由于使用大量的对象,造成很大的存储开销。
对象的大多数状态都可变为外部状态。
如果删除对象的外部状态,那么可以用相对较少的共享对象取代很多组对象。
应用程序不依赖于对象标识。
由于Flyweight对象可以被共享,对于概念上明显有别的对象,标识测试将返回真值。
Proxy
结构 意图为其他对象提供一种代理以控制对这个对象的访问。
适用性在需要用比较通用和复杂的对象指针代替简单的指针的时候,使用Proxy模式。
下面是一些可以使用Proxy模式常见情况:
1)远程代理(RemoteProxy)为一个对象在不同的地址空间提供局部代表。
NEXTSTEP[Add94]使用NXProxy类实现了这一目的。
Coplien[Cop92]称这种代理为“大使”(Ambassador)。
2)虚代理(VirtualProxy)根据需要创建开销很大的对象。
在动机一节描述的ImageProxy就是这样一种代理的例子。
3)保护代理(ProtectionProxy)控制对原始对象的访问。
保护代理用于对象应该有不同的访问权限的时候。
例如,在Choices操作系统[CIRM93]中KemelProxies为操作系统对象提供了访问保护。
4)智能指引(SmartReference)取代了简单的指针,它在访问对象时执行一些附加操作。
它的典型用途包括:
对指向实际对象的引用计数,这样当该对象没有引用时,可以自动释放它(也称为SmartPointers[Ede92])。
当第一次引用一个持久对象时,将它装入内存。
在访问一个实际对象前,检查是否已经锁定了它,以确保其他对象不能改变它。
ChainofResponsibility
结构 意图使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。
将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。
适用性有多个的对象可以
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- C# 中的 设计 模式 原理 doc