美文网首页
装饰者模式

装饰者模式

作者: 闹鬼的金矿 | 来源:发表于2022-05-07 22:44 被阅读0次



    感觉装饰者是一种比较重要的设计模式,它本身是通过组合的思想来实现的。

    先考虑一个简单的例子:比如发送消息给某一个账号,刚开始只有短信一种方式:

    class SMSNotifier {
        func send(String msg) {}
    }
    

    此时新增了两种方式:EMail和微信

    class EMailNotifier {
         func send(String msg){}
    }
    
    class WechatNotifier {
         func send(String msg){}
    }
    

    之后需要同时支持三种中的两种发消息的方式,就需要新增3种新的类型:

    class SMSEMailNotifier {
        func send(String msg){}
    }
    
    class SMSWechatNotifier {
        func send(String msg){}
    }
    
    class EmailWechatNotifier {
        func send(String msg){}
    }
    

    当前这种设计,随着需要支持的通讯类型增加,所需要的类型也会越来越多。并且如果其中一种基本通讯方式发送消息的逻辑发生变化,影响的可能是好几个类。比如Email 变化可能除了Email类本身还会导致其他支持Email的类型(比如EmailWechartNotifier)也需要跟着修改。所以,这种设计对扩展不够友好的。除了现在这种方式,可能还会想到通过继承来实现。但在很多语言中是无法同时继承多个类,并且继承也解决不了类型增加的问题,所以继承的方式也不是特别好的方案。


    如果有一种能够根据实际需求灵活组装类型的方案就好了,同时不需要增加新的类型。此时装饰者就比较合适了,其UML结构如下图所示:

    装饰者模式.png

    上图中Concrete Component只实现了Component接口,它是用来被Decorator进行封装的对象。而Decorator就是用来装饰其他对象的装饰者,同时也可以被其他Decorator装饰。对来调用方Client而言,它只关心Component接口,至于具体内部有多少装饰者对象,并不需要了解。

    装饰者有几个重要的特点:

    1.装饰者里包含了被装饰的对象和装饰者对象,一个对象既可能是被装饰者的对象也可能是装饰者。

    2.2.对接口进行了统一的封装和抽象,被装饰的对象和装饰者对象都需要实现Component接口。对于装饰者而言,可以与任何实现Component接口的对象进行组合。

    3.在装饰者对象中,通过组合的方式引用了被装饰的Component对象。在实际接口excute方法的实现中,它包含装饰者自己的业务逻辑,同时也会调用Component对象的excute方法。通过这种方式,其实是实现了对代码的分层处理。把不同的业务逻辑封装到了不同的装饰者类中。

    4.对于使用方来说,面向的是对同一个接口的依赖,当业务逻辑方式变化的时候,除了装饰者初始化的逻辑发生变化,调用Component excute方法的代码不需要更改。

    1和2实现了可以根据业务需要把不同的装饰者组合起来使用,增加或者减少装饰的层次。


    再看一个实际的例子:在读写一些数据时候,除了数据本身的存储和获取可能还会包含一些其他操作,比如加解密,压缩和解压缩等。

    先创建一个统一的数据业务的抽象接口DataSource,里面有write和read两个基本方法:

    protocol DataSource {
        func read() -> String
        func write(c: String)
    }
    

    FileDataSource是读写文件的类,只包含最基本的IO操作:

    class FileDataSource: DataSource {
        
        var content: String = ""
        
        func read() -> String {
            return content
        }
        
        func write(c: String) {
            content = c
        }
    }
    

    现在创建一个装饰者的基类,他比FileDataSource多了一个实现DataSource接口类型的字段。它本身被用做其他具体装饰者的基类,所以read和write方法没有其他内容,直接调用ds的read和write方法:

    class DataSourceDecorator: DataSource {
        
        var ds: DataSource
        
        init(data: DataSource) {
            ds = data
        }
        
        func read() -> String {
            return ds.read()
        }
        
        func write(c: String) {
            ds.write(c: c)
        }
    }
    

    接着创建两个具体的装饰者对象:用于加解密,压缩和解压缩的装饰者,它们在读取数据之后进行解密和解压缩,在存储数据之前进行加密和压缩。

    class EncryptionDecorator: DataSourceDecorator {
        
        override func read() -> String {
            let s = self.ds.read()
            print("解密...")
            return s
        }
        
        override func write(c: String) {
            print("加密...")
            self.ds.write(c: c)
        }
    }
    
    class CompressionDecorator: DataSourceDecorator {
        
        override func read() -> String {
            let c = self.ds.read()
            print("解压缩...")
            return c
        }
        
        override func write(c: String) {
            print("压缩...")
            self.ds.write(c: c)
        }
    }
    

    测试代码,能够看到加入不同的装饰者之后的变化:

    func testDecorator() {
        
        var d:DataSource = FileDataSource()
        d.write(c: "测试数据")
        print("\(d.read())")
        
        print("\n")
        
        d = EncryptionDecorator(data: d)
        d.write(c: "测试数据")
        print("\(d.read())")
        
        print("\n")
        
        d = CompressionDecorator(data: d)
        d.write(c: "测试数据")
        print("\(d.read())")
        
    }
    

    Reference: Dive Into DESIGN PATTERNS

    相关文章

      网友评论

          本文标题:装饰者模式

          本文链接:https://www.haomeiwen.com/subject/gvkfurtx.html