实例讲解Ruby使用设计模式中的装饰器模式的方法

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

解决方案

  • 动态地给一个对象添加一些额外的职责或者行为。就增加功能来说, Decorator模式相比生成子类更为灵活。
  • 提供了改变子类的灵活方案。装饰器模式在不必改变原类文件和使用继承的情况下,动态的扩展一个对象的功能。它是通过创建一个包装对象,也就是装饰来包裹真实的对象。
  • 当用于一组子类时,装饰器模式更加有用。如果你拥有一族子类(从一个父类派生而来),你需要在与子类独立使用情况下添加额外的特性,你可以使用装饰器模式,以避免代码重复和具体子类数量的增加。

 适用性

以下情况使用Decorator模式

  • 在不影响其他对象的情况下,以动态、透明的方式给单个对象添加职责。
  • 处理那些可以撤消的职责。
  • 当不能采用生成子类的方法进行扩充时。一种情况是,可能有大量独立的扩展,

为支持每一种组合将产生大量的子类,使得子类数目呈爆炸性增长。

另一种情况可能是因为类定义被隐藏,或类定义不能用于生成子类。
 
实例

class SimpleWriter
 def initialize(path)
  @file = File.open(path,"w")
 end 

 def write_line(line)
  @file.print(line)
  @file.print("\n")
 end 

 #字符数
 def pos
  @file.pos
 end 

 #它将会将文件指针指向文件的开头
 def rewind
  @file.rewind
 end 

 def close
  @file.colse
 end
end 

sw = SimpleWriter.new("test.txt")
sw.write_line("你好")
puts sw.pos
puts sw.rewind 

#基类
class WriterDecorator
 def initialize(real_writer)
  @real_writer = real_writer
 end 

 def write_line
  @real_writer.write_line
 end 

 def pos
  @real_writer.pos
 end 

 def rewind
  @real_writer.rewind
 end 

 def close
  @real_writer.close
 end
end 

class NumberingWriter < WriterDecorator
 attr :line_number
 def initialize(real_writer)
  super(real_writer)
  @line_number = 1
 end 

 #实际调用的是WriterDecorator的write_line方法,只是在写入的内容前加上了编号(装饰)
 #所以说NumberingWriter对WriterDecorator的接口wirte_line进行了装饰
 #
 def write_line(line)
  @real_writer.write_line("#{@line_number}:#{line}")
  @line_number += 1
 end
end 

sw = SimpleWriter.new("numbering_write.txt")
nw = NumberingWriter.new(sw)
nw.write_line("hello,world")
nw.write_line("hello,ruby")
puts nw.line_number 

class CheckSummingWriter < WriterDecorator
 attr_reader :check_num 

 def initialize(real_writer)
  super(real_writer)
  @check_num = 0
 end 

 def write_line(line)
  line.each_byte{|byte| @check_num += byte % 256}
  @real_writer.write_line(line)
 end
end 

sw = SimpleWriter.new("check_num_writer.txt")
csw = CheckSummingWriter.new(sw)
csw.write_line("hello,world")
puts csw.check_num 

class TimeStampingWriter < WriterDecorator 

 def initialize(real_writer)
  super(real_writer)
 end 

 def write_line(line)
  @real_writer.write_line("#{Time.now}: #{line}")
 end 

end 

#倒着看
#5. 实际调用的是SimpleWriter得write_line方法,将内容写入文件
sw = SimpleWriter.new("mix.txt")
#4. 实际调用的是NumberingWriter得write_line方法,对在输入的数据前加上了编号
#  然后传给@real_writer,此时的@real_witer为sw
nw = NumberingWriter.new(sw)
#3. 实际调用的是TimeStampingWriter得write_line方法,对在输入的数据前加上了时间戳
#  然后传给@real_writer,此时的@real_witer为nw
tsw = TimeStampingWriter.new(nw)
#2. 实际调用的是CheckSummingWriter得write_line方法,对输入的数据进行了字节数的统计
#  然后传给@real_writer,此时的@real_witer为tsw
csw = CheckSummingWriter.new(tsw)
#1. csw调用write_line
csw.write_line("hello,world")
puts csw.check_num

两种ruby风格的装饰模式应用
 
(1)使用extend混入模块

class SimpleWriter
 def initialize(path)
  @file = File.open(path,"w")
 end 

 def write_line(line)
  @file.print(line)
  @file.print("\n")
 end 

 #字符数
 def pos
  @file.pos
 end 

 #它将会将文件指针指向文件的开头
 def rewind
  @file.rewind
 end 

 def close
  @file.colse
 end
end 

#使用extend方法动态的混入模块,来进行装饰 

module TimeStampingWriter
 def write_line(line)
  super("#{Time.now}:#{line}")
 end
end 

module NumberingWriter
 attr_reader :line_number
 def write_line(line)
  @line_number = 1 unless @line_number
  super("#{@line_number}:#{line}")
  @line_number += 1
 end
end

最后被加入的模块,先被调用,然后通过super来调用父类的write_line方法。 
例子中先在文本的前面加上时间戳,在加入编号,最后写入文件

sw = SimpleWriter.new("out3.txt")
sw.extend(NumberingWriter)
sw.extend(TimeStampingWriter)
sw.write_line("hello,ruby")

(2)使用alias关键字

class SimpleWriter
 def initialize(path)
  @file = File.open(path,"w")
 end 

 def write_line(line)
  @file.print(line)
  @file.print("\n")
 end 

 #字符数
 def pos
  @file.pos
 end 

 #它将会将文件指针指向文件的开头
 def rewind
  @file.rewind
 end 

 def close
  @file.colse
 end
end 

ruby实现装饰模式的另一种动态方法 :
修改对象的实例方法, 所以在out1.txt文件中会加入时间戳,而不影响对象sw2,out2.txt中不会加入时间戳 。
sw1 = SimpleWriter.new("out1.txt")
class << sw1
 alias old_write_line write_line
 def write_line(line)
  old_write_line("#{Time.now}:#{line}")
 end
end
sw1.write_line("hello,world")
sw2 = SimpleWriter.new("out2.txt")
sw2.write_line("hello,world")
(0)

相关推荐

  • PHP、Python和Javascript的装饰器模式对比

    修饰模式(Decorator Pattern),又叫装饰者模式,是面向对象编程领域中,一种动态地往一个类中添加新的行为的设计模式.就功能而言,修饰模式相比生成子类更为灵活,这样可以给某个对象而不是整个类添加一些功能.装饰模式非常适用于灵活扩展对象的功能,下面是装饰模式的UML图: 例如,有一个技术论坛,用户通过留言进行沟通,由于刚开始论坛里都是熟人,几乎都不需要对留言的内容作出审核,接收留言的页面可以是这样: class SaveMsg(){ private $msg; public funct

  • 深入解析设计模式中的装饰器模式在iOS应用开发中的实现

    装饰器模式可以在不修改代码的情况下灵活的为一对象添加行为和职责.当你要修改一个被其它类包含的类的行为时,它可以代替子类化方法. 一.基本实现 下面我把类的结构图向大家展示如下: 让我们简单分析一下上面的结构图,Component是定义一个对象接口,可以给这些对象动态地添加职责.ConcreteComponent是定义了一个具体的对象,也可以给这个对象添加一些职责.Decorator,装饰抽象类,继承了Component,从外类来扩展Component类的功能,但对于Component来说,是无需

  • C#装饰器模式(Decorator Pattern)实例教程

    本文以实例形式详细讲述了C#装饰器模式的实现方法.分享给大家供大家参考.具体实现方法如下: 现假设有一个公司要做产品套餐,即把不同的产品组合在一起,不同的组合对应不同的价格.最终呈现出来的效果是:把产品组合的所有元素呈现出来,并显示该组合的价格. 每个产品都有名称和价格,首先设计一个关于产品的抽象基类. public abstract class ProductBase { public abstract string GetName(); public abstract double GetP

  • 学习php设计模式 php实现装饰器模式(decorator)

    动态的给一个对象添加一些额外的职责.就增加功能来说,Decorator模式相比生成子类更为灵活[GOF95] 装饰模式是以对客户透明的方式动态地给一个对象附加上更多的职责.这也就是说,客户端并不会觉得对象在装饰前和装饰后有什么不同.装饰模式可以在不使用创造更多子类的情况下,将对象的功能加以扩展. 一.装饰模式结构图 二.装饰模式中主要角色 抽象构件(Component)角色:定义一个对象接口,以规范准备接收附加职责的对象,从而可以给这些对象动态地添加职责. 具体构件(Concrete Compo

  • Python的装饰器模式与面向切面编程详解

    今天来讨论一下装饰器.装饰器是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理等.装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量函数中与函数功能本身无关的雷同代码并继续重用.概括的讲,装饰器的作用就是为已经存在的对象添加额外的功能. 1. 装饰器入门 1.1. 需求是怎么来的? 装饰器的定义很是抽象,我们来看一个小例子. 复制代码 代码如下: def foo():     print 'in foo()'   foo() 这是一个很无聊

  • java设计模式之桥接模式(Bridge)

    概述 桥接模式一种结构型模式,它主要应对的是:由于实际的需要,某个类具有两个或以上的维度变化,如果只是用继承将无法实现这种需要,或者使得设计变得相当臃肿. 桥接模式的做法是把变化的部分抽象出来,使变化部分与主类分离开来,从而将多个维度的变化彻底分离.最后,提供一个管理类来组合不同维度上的变化,通过这种组合来满足业务的需要. UML结构图 代码示例 package interview; interface Implementor{ void operationImpl(); } abstract

  • 分析Python中设计模式之Decorator装饰器模式的要点

    先给出一个四人团对Decorator mode的定义:动态地给一个对象添加一些额外的职责. 再来说说这个模式的好处:认证,权限检查,记日志,检查参数,加锁,等等等等,这些功能和系统业务无关,但又是系统所必须的,说的更明白一点,就是面向方面的编程(AOP). 在Python中Decorator mode可以按照像其它编程语言如C++, Java等的样子来实现,但是Python在应用装饰概念方面的能力上远不止于此,Python提供了一个语法和一个编程特性来加强这方面的功能.Python提供的语法就是

  • java设计模式之组合模式(Composite)

    概述 是一种结构型模式,将对象以树形结构组织起来,以表示"部分 - 整体"的层次结构,使得客户端对单个对象和组合对象的使用具有唯一性. UML类图 上面的类图包含的角色: Component:为参加组合的对象声明一个公共的接口,不管是组合还是叶节点. Leaf:在组合中表示叶子结点对象,叶子结点没有子结点. Composite:表示参加组合的有子对象的对象,并给出树枝构建的行为: 代码示例 import java.util.ArrayList; import java.util.Lis

  • java设计模式之观察者模式

    观察者模式又称发布-订阅(Publish/Subscribe)模式,定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象.这个主题对象在状态发生变化时,会通知所有观察者对象,使他们能够自动更新自己.将一个系统分割成一系列相互协作的类有一个很不好的副作用,那就是需要维护相关对象间的一致性.我们不希望为了维持一致性而使各类紧密耦合,这样会给维护.扩展和复用都带来不便.观察者模式所做的工作其实就是在解除耦合,让耦合的双方都依赖于抽象,而不是依赖于具体. 观察者模式是实际中应用比较广泛的模

  • java设计模式之装饰器模式(Decorator)

    概述 装饰模式是对客户端以透明的方式扩展对象的功能,是继承关系的一个替代方案.也就是说,客户端并不会觉得对象在装饰前和装饰后有什么不同,装饰模式可以在不用创造更多子类的情况下,将对象的功能加以扩展,装饰模式的关键在于这种扩展是完全透明的. 模式的结构 UML类图: 装饰模式中的类角色: 抽象构件角色(Project):给出一个接口,以规范准备接收附加责任的对象 具体构件角色(Employe):定义一个将要接收附加责任的类 装饰角色(Manager):持有一个构件对象的实例,并定义一个与抽象构件接

随机推荐