设计模式

3、装饰模式

2017-07-13  本文已影响14人  barry

1. 概述

      若你从事过面向对象开发,实现给一个类或对象增加行为,使用继承机制,这是所有面向对象语言的一个基本特性。如果已经存在的一个类缺少某些方法,或者须要给方法添加更多的功能(魅力),你也许会仅仅继承这个类来产生一个新类—这建立在额外的代码上。
      通过继承一个现有类可以使得子类在拥有自身方法的同时还拥有父类的方法。但是这种方法是静态的,用户不能控制增加行为的方式和时机。如果 你希望改变一个已经初始化的对象的行为,你怎么办?或者,你希望继承许多类的行为,该怎么办?前一个,只能在于运行时完成,后者显然时可能的,但是可能会导致产生大量的不同的类—可怕的事情。

2. 问题

      你如何组织你的代码使其可以容易的添加基本的或者一些很少用到的 特性,而不是直接添加额外的代码写在你的类的内部?

3. 解决方案

       装饰器模式: 动态地给一个对象添加一些额外的职责或者行为。就增加功能来说, Decorator模式相比生成子类更为灵活。

      装饰器模式提供了改变子类的灵活方案。装饰器模式在不必改变原类文件和使用继承的情况下,动态的扩展一个对象的功能。它是通过创建一个包装对象,也就是装饰来包裹真实的对象。

      当用于一组子类时,装饰器模式更加有用。如果你拥有一族子类(从一个父类派生而来),你需要在与子类独立使用情况下添加额外的特性,你可以使用装饰器模式,以避免代码重复和具体子类数量的增加。

4. 适用性

以下情况使用Decorator模式

5. 结构

uml如图:

装饰模式.png

抽象基类

//抽象基类
class Component: NSObject {

    func show() {
        
    }
}

人类

//人类(被装饰对象)
class Person: Component {

    private var name: String;
    
    init(name: String) {
        self.name = name
        
        super.init()
    }
    
    override func show() {
        print("装扮的\(name)")
    }
}

服饰基类

//服饰基类
class Finery: Component {
    
    private var component: Component?
    
    func decorate(component: Component) {
        self.component = component
    }
    
    override func show() {
        
        guard let component = component else {
            return
        }
        
        component.show()
    }
}

服饰子类(其它子类类似)

//大T恤
class TShirts: Finery {

    override func show() {
        print("大T恤")
        
        super.show()
    }
}


9 实现2

因为只有一个具体被装饰类,所以可以除去抽象基类Component,以Person类作为抽象基类

(抽象基类)

//人类(抽象基类)
class Person: NSObject {

    private var name: String;
    
    init(name: String) {
        self.name = name
        
        super.init()
    }
    
    func show() {
        print("装扮的\(name)")
    }
}

服饰基类(服饰子类不作改动)

//服饰基类
class Finery: Person {
    
    private var person: Person?
    
    init() {
        super.init(name: "")
    }
    
    func decorate(person: Person) {
        self.person = person
    }
    
    override func show() {
        
        guard let person = person else {
            return
        }
        
        person.show()
    }
}

Github传送门

上一篇 下一篇

猜你喜欢

热点阅读