Ruby设计模式之策略模式一: 基本的策略模式

时间:2022-09-09 09:28:25

模板模式固然不错,但其还是有一些缺陷的。比如其实现依赖于继承并且缺足够的灵活性。在这时候我们就需要找到一个更加优化的解决方案——策略模式。

下面是使用策略模式实现的Report模板

 1  # encoding: utf-8
 2 
 3  class  Formatter
 4       def  output_report title, text
 5           raise   ' can not call Abstract method '
 6      end
 7  end
 8 
 9  class  HTMLFormatter  <  Formatter
10       def  output_report title, text
11          puts  ' <html> '
12          puts  '     <head> '
13          puts  '         <title> '   +  title  +   ' </title> '
14          puts  '     </head> '
15          puts  '     <body> '
16          text.each do  | line |
17              puts  " <p>#{line}</p> "
18          end
19          puts  '     </body> '
20          puts  ' </html> '
21      end
22  end
23 
24  class  PlainTextFormatter  <  Formatter
25       def  output_report title, text
26          puts  ' ********  '   +  title  +   '  ******** '
27          text.each do  | line |
28              puts line
29          end
30      end
31  end
32 
33  class  Reporter
34      attr_reader :title, :text
35      attr_accessor :formater
36 
37       def  initialize formater
38          @title  =   ' My Report '
39          @text  =  [ ' This is my report ' ' Please see the report ' ' It is ok ' ]
40          @formater  =  formater
41      end
42 
43       def  output_report
44          @formater.output_report @title, @text
45      end
46  end
47 
48  Reporter.new(HTMLFormatter.new).output_report
49  Reporter.new(PlainTextFormatter.new).output_report

GOF将Strategy模式称之为pull the algorithm out into a separate object,其基本理念就是实现一系列strategies对象,这些对象全都实现相同的功能。在上面的例子中这些strategies对象的功能就是输入报告。

Strategies对象除了实现相同功能之外,还需要定义统一的接口。上面的例子里就是实现了output_report方法。

使用这些Strategies的类被GOF称之为context。context可以调用任意的strategies,在context看来这些strategies并没有什么不一样,因为它们都实现了同样的功能。不过选择不同的strategies还是有些许不同的。在上面的例子里,选择HTMLFormatter可以输出HTML报告,而选择PlainTextFormatter则输入另一种风格的报告。如果我们实现一个住房公积金计算器,那么我们可以实现多个不同的strategy,1个strategy用来实现深圳的公积金算法,1个实现上海的算法,1个实现北京的算法如此类推。

策略模式有如下的优点

  • 策略模式可以分离不同算法的代码,而且context不需要知道任何有关strategy的实现细节,其只需要调用strategy所定义的接口既可。

  • 另外在context中,切换strategy的成本低廉且灵活。