脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Ruby - 实例讲解Ruby使用设计模式中的装饰器模式的方法

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

2020-05-07 11:13michael_roshen Ruby

这篇文章主要介绍了Ruby使用设计模式中的装饰器模式的实例, 装饰模式能够实现动态的为对象添加功能,需要的朋友可以参考下

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

解决方案

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

 适用性

       以下情况使用Decorator模式

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

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

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
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混入模块

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
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方法。 
例子中先在文本的前面加上时间戳,在加入编号,最后写入文件  

?
1
2
3
4
sw = SimpleWriter.new("out3.txt")
sw.extend(NumberingWriter)
sw.extend(TimeStampingWriter)
sw.write_line("hello,ruby")

 
(2)使用alias关键字

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
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")

延伸 · 阅读

精彩推荐
  • Ruby简要说明Ruby中的迭代器

    简要说明Ruby中的迭代器

    这篇文章主要介绍了Ruby中的迭代器,迭代器的概念在动态语言的编程中十分重要,文章中介绍了Ruby中的each迭代器和collect迭代器,需要的朋友可以参考下 ...

    goldensun2772020-04-25
  • RubyRuby进行文件信息输出实例代码

    Ruby进行文件信息输出实例代码

    Ruby进行文件信息输出实例代码,数据是随机的,所以每次的记录都会不同。 ...

    ruby教程网2962020-04-10
  • RubyRuby简洁学习笔记(一):字符串、数字、类和对象

    Ruby简洁学习笔记(一):字符串、数字、类和对象

    这篇文章主要介绍了Ruby简洁学习笔记(一):字符串、数字、类和对象,本文是学习笔记第一篇,需要的朋友可以参考下 ...

    脚本之家2472020-04-20
  • RubyRuby迭代器的7种技巧分享

    Ruby迭代器的7种技巧分享

    这篇文章主要介绍了Ruby迭代器的7种技巧分享,Ruby中的迭代器非常人性化,本文既是讲解了7个技巧也是讲解了7种迭代器,需要的朋友可以参考下 ...

    脚本之家4782020-04-20
  • Ruby剖析 Ruby 访问控制

    剖析 Ruby 访问控制

    前面,我们说 Ruby 没有函数,只有方法.而且实际上有不止一种方法.这一节我们介绍 访问控制 (accesscontrols). 想想当我们在最高层而不是在一个类的定义里定义...

    ruby教程网3572020-04-08
  • RubyCentOS中配置Ruby on Rails环境

    CentOS中配置Ruby on Rails环境

    经过一个上午的折腾,终于把ROR环境在CentOS中搞定,绕了很多弯路,把文章写下来总结一下 ...

    可乐加糖4762020-04-12
  • RubyRuby设计模式编程中使用Builder建造者模式的实例

    Ruby设计模式编程中使用Builder建造者模式的实例

    这篇文章主要介绍了Ruby设计模式编程中使用Builder建造者模式的实例,建造者模式将一个复杂对象的构造与它的表示分离,使同样的构建过程可以创建不同的表...

    范孝鹏2192020-05-07
  • RubyRuby环境下安装使用bundler来管理多版本的gem

    Ruby环境下安装使用bundler来管理多版本的gem

    这篇文章主要介绍了Ruby环境下安装使用bundler来管理多版本的gem的方法,举了Ruby On Rails中的应用实例来进行演示,需要的朋友可以参考下 ...

    日拱一卒4332020-05-10