结构型设计概述和适配器模式.ppt
结构型设计模式,桂勋,结构型设计模式,结构型设计模式概述,结构型模式(Structural Pattern)描述如何将类或者对象结合在一起形成更大的结构,就像搭积木,可以通过简单积木的组合形成复杂的、功能更为强大的结构。,结构型设计模式的概述,结构型模式可以分为类结构型模式和对象结构型模式:类结构型模式关心类的组合,由多个类可以组合成一个更大的系统,在类结构型模式中一般只存在继承关系和实现关系。对象结构型模式关心类与对象的组合,通过关联关系使得在一个类中定义另一个类的实例对象,然后通过该对象调用其方法。根据“合成复用原则”,在系统中尽量使用关联关系来替代继承关系,因此大部分结构型模式都是对象结构型模式。,GOF95中的特化图(1),1、类图,2、对象图,3、参与客户和绝对客户,GOF95中的特化图(2),GOF95中的特化图(3),第一个请求是aCreationTool发出的,请求创建aLineShape。接下来,aLineShape被加入到aDrawing中,这导致aDrawing向它自身发出一个Refresh请求。而在Refresh操作过程中aDrawing又向aLineShape发出一个Draw请求。,Adapter-适配器Bridge-桥接Composite-组合Decorator-装饰Facade-外观Flyweight-享元Proxy-代理,结构型设计模式,Adapter-适配器,主要内容,模式动机模式定义模式结构模式的程序表示C+模式的程序表示Java实例与解析模式效果评价模式的适用环境模式扩展已知应用模式小结,Adapter-适配器:动机,Adapter-适配器:动机,在软件开发中采用类似于电源适配器的设计和编码技巧被称为适配器模式。通常情况下,客户端可以通过目标类的接口访问它所提供的服务。有时,现有的类可以满足客户类的功能需要,但是它所提供的接口不一定是客户类所期望的,这可能是因为现有类中方法名与目标类中定义的方法名不一致等原因所导致的。在这种情况下,现有的接口需要转化为客户类期望的接口,这样保证了对现有类的重用。如果不进行这样的转化,客户类就不能利用现有类所提供的功能,适配器模式可以完成这样的转化。,Adapter-适配器:动机,在适配器模式中可以定义一个包装类,包装不兼容接口的对象,这个包装类指的就是适配器(Adapter),它所包装的对象就是适配者(Adaptee),即被适配的类。适配器提供客户类需要的接口,适配器的实现就是把客户类的请求转化为对适配者的相应接口的调用。也就是说:当客户类调用适配器的方法时,在适配器类的内部将调用适配者类的方法,而这个过程对客户类是透明的,客户类并不直接访问适配者类。因此,适配器可以使由于接口不兼容而不能交互的类可以一起工作。这就是适配器模式的模式动机。,Adapter-适配器:动机,GOF95中的例子:,Adapter-适配器:定义,适配器模式(Adapter Pattern):将一个接口转换成客户希望的另一个接口,适配器模式使接口不兼容的那些类可以一起工作,其别名为包装器(Wrapper)。适配器模式既可以作为类结构型模式,也可以作为对象结构型模式。,Adapter-适配器:结构,(1)类适配器:类适配器使用多重继承对一个接口与另一个接口进行匹配,Adapter-适配器:结构,(2)对象适配器:对象适配器依赖于对象组合,Adapter-适配器:结构,Target(Shape)定义Client使用的与特定领域相关的接口。Client(DrawingEditor)与符合Target接口的对象协同。Adaptee(TextView)定义一个已经存在的接口,这个接口需要适配。Adapter(TextShape)对Adaptee的接口与Target接口进行适配,Adapter-适配器:程序表示C+,class Shape public:Shape();virtual void BoundingBox(Point,class TextView public:TextView();void GetOrigin(Coord,class Manipulator;class TextManipulator:public Manipulator public:TextManipulator(const TextShape*);,Adapter-适配器:程序表示C+,(1)类适配器,class TextShape:public Shape,private TextView public:TextShape();virtual void BoundingBox(Point,void TextShape:BoundingBox(Point,Adapter-适配器:程序表示C+,(1)对象适配器,TextShape:TextShape(TextView*t)_text=t;void TextShape:BoundingBox(Point,class TextShape:public Shape public:TextShape(TextView*);virtual void BoundingBox(Point,Adapter-适配器:程序表示Java,Java中典型的类适配器代码:,Adapter-适配器:程序表示Java,Java中典型的对象适配器代码:,Adapter-适配器:实例与解析,实例一:仿生机器人 现需要设计一个可以模拟各种动物行为的机器人,在机器人中定义了一系列方法,如机器人叫喊方法cry()、机器人移动方法move()等。如果希望在不修改已有代码的基础上使得机器人能够像狗一样叫,像狗一样跑,使用适配器模式进行系统设计。,Adapter-适配器:实例与解析,实例二:加密适配器某系统需要提供一个加密模块,将用户信息(如密码等机密信息)加密之后再存储在数据库中,系统已经定义好了数据库操作类。为了提高开发效率,现需要重用已有的加密算法,这些算法封装在一些由第三方提供的类中,有些甚至没有源代码。使用适配器模式设计该加密模块,实现在不修改现有类的基础上重用第三方加密方法。,Adapter-适配器:效果评价,模式优缺点适配器模式的优点如下:将目标类和适配者类解耦,通过引入一个适配器类来重用现有的适配者类,而无需修改原有代码。增加了类的透明性和复用性,将具体的实现封装在适配者类中,对于客户端类来说是透明的,而且提高了适配者的复用性。灵活性和扩展性都非常好,通过使用配置文件,可以很方便地更换适配器,也可以在不修改原有代码的基础上增加新的适配器类,完全符合“开闭原则”。,Adapter-适配器:效果评价,模式优缺点类适配器模式还具有如下优点:由于适配器类是适配者类的子类,因此可以在适配器类中置换一些适配者的方法,使得适配器的灵活性更强。类适配器模式的缺点如下:对于Java、C#等不支持多重继承的语言,一次最多只能适配一个适配者类,而且目标抽象类只能为抽象类,不能为具体类,其使用有一定的局限性,不能将一个适配者类和它的子类都适配到目标接口。,Adapter-适配器:适用环境,系统需要使用现有的类,而这些类的接口不符合系统的需要。想要建立一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。,Adapter-适配器:模式扩展,缺省适配器模式(Default Adapter Pattern)当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆盖父类的某些方法来实现需求,它适用于一个接口不想使用其所有的方法的情况。因此也称为单接口适配器模式。,Adapter-适配器:模式扩展,缺省适配器模式适配者接口 缺省适配器类 具体业务类,Adapter-适配器:模式扩展,缺省适配器模式举例(1)宽接口:,Adapter-适配器:模式扩展,缺省适配器模式举例(2)【GOF95】【P95】从已有接口中寻找最小适配接口(窄接口)。,Adapter-适配器:模式扩展,缺省适配器模式举例(2)【GOF95】【P96】从已有接口中寻找最小适配接口,采用独立抽象并结合Proxy代理模式。,Adapter-适配器:模式扩展,双向适配器 在对象适配器的使用过程中,如果在适配器中同时包含对目标类和适配者类的引用,适配者可以通过它调用目标类中的方法,目标类也可以通过它调用适配者类中的方法,那么该适配器就是一个双向适配器。,Adapter-适配器:模式扩展,双向适配器结构:,Adapter-适配器:模式扩展,双向适配器模式举例【GOF95】【P95】:考虑一个双向适配器,它将图形编辑框架Unidraw与约束求解工具箱QOCA集成起来。这两个系统都有一些类,这些类显式地表示变量:Unidraw含有类StateVariable,QOCA中含有类ConstraintVariable,如下图所示。为了使Unidraw与QOCA协同工作,必须首先使类ConstraintVariable与类StateVariable相匹配;而为了将QOCA的求解结果传递给Unidraw,必须使StateVariable与ConstraintVariable相匹配,Adapter-适配器:已知应用,(1)Sun公司在1996年公开了Java语言的数据库连接工具JDBC,JDBC使得Java语言程序能够与数据库连接,并使用SQL语言来查询和操作数据。JDBC给出一个客户端通用的抽象接口,每一个具体数据库引擎(如SQL Server、Oracle、MySQL等)的JDBC驱动软件都是一个介于JDBC接口和数据库引擎接口之间的适配器软件。抽象的JDBC接口和各个数据库引擎API之间都需要相应的适配器软件,这就是为各个不同数据库引擎准备的驱动程序。(2)在组件对象模型COM中,通过聚合使新开发的组件能集成已有的组件,新组件就变成了一个用户适配器(二进制适配器)。,Adapter-适配器:已知应用,(3)在Spring AOP框架中,对BeforeAdvice、AfterAdvice、ThrowsAdvice三种通知类型借助适配器模式来实现。,Adapter-适配器:已知应用,(4)在JDK类库中也定义了一系列适配器类,如在包中定义的InputStreamAdapter类,用于包装ImageInputStream接口及其子类对象。,Adapter-适配器:模式小结,结构型模式描述如何将类或者对象结合在一起形成更大的结构。适配器模式用于将一个接口转换成客户希望的另一个接口,适配器模式使接口不兼容的那些类可以一起工作,其别名为包装器。适配器模式既可以作为类结构型模式,也可以作为对象结构型模式。适配器模式包含四个角色:目标抽象类定义客户要用的特定领域的接口;适配器类可以调用另一个接口,作为一个转换器,对适配者和抽象目标类进行适配,它是适配器模式的核心;适配者类是被适配的角色,它定义了一个已经存在的接口,这个接口需要适配;在客户类中针对目标抽象类进行编程,调用在目标抽象类中定义的业务方法。在类适配器模式中,适配器类实现了目标抽象类接口并继承了适配者类,并在目标抽象类的实现方法中调用所继承的适配者类的方法;在对象适配器模式中,适配器类继承了目标抽象类并定义了一个适配者类的对象实例,在所继承的目标抽象类方法中调用适配者类的相应业务方法。,Adapter-适配器:模式小结,适配器模式的主要优点是将目标类和适配者类解耦,增加了类的透明性和复用性,同时系统的灵活性和扩展性都非常好,更换适配器或者增加新的适配器都非常方便,符合“开闭原则”;类适配器模式的缺点是适配器类在很多编程语言中不能同时适配多个适配者类,对象适配器模式的缺点是很难置换适配者类的方法。适配器模式适用情况包括:系统需要使用现有的类,而这些类的接口不符合系统的需要;想要建立一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类一起工作。,