现在的位置: 首页 > 综合 > 正文

2.策略模式(下)

2016年02月27日 ⁄ 综合 ⁄ 共 4092字 ⁄ 字号 评论关闭

策略模式(上) 中,留下了一些练习。通过这些练习可以了解Java API、SWT如何实际应用策略模式。下篇将给出这些练习的参考分析,欢迎大家一起讨论。

#1.请说明策略模式符合哪些面向对象设计原则?

让我们再回顾一下策略模式:

  •  策略模式中,通过接口(interface)实现了业务与具体实现的隔离。所以,策略模式符合“面向接口,而不是面向实现”的编程原则。
  • 在实现具体的策略(算法)时,通过实现接口可以方便地增加多个具体实现。所以,策略模式符合“开放封闭原则,对扩展开放,对修改封闭”。
  • 业务代码(Context)只依赖规则接口(Strategy),不依赖具体实现。所以,策略模式符合“依赖倒置原则,高层(业务模块)不依赖底层模块(具体算法),只依赖抽象接口”。

#2.请说明策略模式有什么优点和缺点。

  • 优点:遵循面向对象设计原则,降低设计耦合、便于扩展。
  • 缺点:当实现算法较多时,会增加需要维护的类的数量。可以使用工厂方法来解决。

#3.请用说明下列Java API设计中是如何使用策略模式的?

Java实例1. 集合、数组排序(sort)

实际编程中最常见的莫过于排序了。在Java API中,Collections.sort是常用方法之一(JDK7代码):

1
2
3
4
5
6
7
8
9
public
static
<T> void
sort(List<T> list, Comparator<? super
T> c) {
    Object[] a = list.toArray();
    Arrays.sort(a, (Comparator)c);
    ListIterator i = list.listIterator();
    for
(int
j=
0; j<a.length; j++) {
        i.next();
        i.set(a[j]);
    }
}

这里调用了Arrays.sort

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public
static
<T> void
sort(T[] a, Comparator<? super
T> c) {
    if
(LegacyMergeSort.userRequested)
        legacyMergeSort(a, c);
    else
        TimSort.sort(a, c);
}
 
/** To be removed in a future release. */
private
static
<T> void
legacyMergeSort(T[] a, Comparator<? super
T> c) {
    T[] aux = a.clone();
    if
(c==null)
        mergeSort(aux, a,
0, a.length,
0);
    else
        mergeSort(aux, a,
0, a.length,
0, c);
}

这里是如何运用策略模式的呢?

  • Context:是实际调用排序的代码,即Collections的类。可以看到,默认使用了TimSort进行排序,排序的算法与对象的compare具体实现无关。
  • Strategy:即具体的对象比较接口,Comparator。
  • ConcreteStrategy:具体的比较算法,通过Comparator实现。例如实现大小写敏感、大小写不敏感或任意规则的比对。

Java实例2.正则表达式

正则表达式也是日常开发常用的工具,让我们来看看Pattern, Matcher是如何运用策略模式的。下面以一段简单的正则调用代码为例:

1
2
3
4
5
6
7
8
Pattern pattern = Pattern.compile("ab", Pattern.CASE_INSENSITIVE);
Matcher matcher = pattern.matcher("ABcabdAb");
// using Matcher find(), group(), start() and end() methods
while
(matcher.find()) {
    System.out.println("Found the text \""
+ matcher.group()
            +
"\" starting at "
+ matcher.start()
            +
" index and ending at index "
+ matcher.end());
}

可以看到,上面的示例中对字符串”ABcabdAb”用”ab”进行大小写不敏感匹配。

  • Context:是实际调用正则表达式匹配的代码,即Matcher类。可以看到,Matcher.find()实际执行了正则匹配并返回匹配结果。
  • Strategy:即匹配的接口。这里并没有使用Java的interface,而是通过Pattern构造函数进行了传递,只要填入了Pattern(String, int),正则表达式和匹配参数,即实现了匹配需要的接口。
  • ConcreteStrategy:具体的匹配算法,pattern字符串需要符合Java正则表达式的语法规则,而匹配的参数flags由Pattern中的常量定义。

Java实例3.线程池

在Java中,线程池的使用也十分常见。让我们看下面这段简单的示例:

1
2
3
4
5
6
7
8
9
10
ThreadPoolExecutor threadPool =
new ThreadPoolExecutor(3,
5, 60,
        TimeUnit.SECONDS,
new ArrayBlockingQueue<Runnable>(10),
        new
ThreadPoolExecutor.CallerRunsPolicy());
for
(
int i =
0; i < 10; i++) {
    System.out.println("add job_"
+ i + " at:"
+
new Date());
    SimplePrintJob job =
new SimplePrintJob("job_"
+ i);
    threadPool.execute(job);
}
System.out.println("execute done!");
threadPool.shutdown();

在构造ThreadPoolExecutor时使用了两个参数,排队策略和拒绝任务的处理策略。其中排队策略可以方便地扩展,比如实现自定义策略。而任务的拒绝处理策略。

  • Context:ThreadPoolExecutor实际执行的方法execute()。按照用户设定的排队策略和拒绝任务的处理策略执行。
  • Strategy:这里排队策略和拒绝任务的处理策略提供了各自的接口,分别是BlockingQueue<E>和RejectedExecutionHandler。
  • ConcreteStrategy:具体的排队策略和拒绝任务的处理策略,JDK中都提供了默认的实现方法。可以实现接口自定义策略进行扩展。

Java实例4.ForkJoin框架

Java7中引入了ForkJoin框架,对多线程开发带来了很大的便利。关于ForkJoin框架的介绍可参见这篇文章

ForkJoin框架通过线程并发实现了对可以分治的问题提供了统一的抽象(业务)。开发者只需继承RecursiveAction或RecursiveTask,实现对应的abstract方法,就可以高效地实现问题求解。

下面让我们来看看ForkJoin框架是如何应用策略模式:

  • Context:ForkJoinPool实际执行的方法invoke()。按照用户提供的算法实现并发执行任务。
  • Strategy:ForkJoin框架通过RecursiveTask、RecursiveAction抽象类提供策略接口。
  • ConcreteStrategy:具体的实现,比如计算斐波那契数列、排序等,可以在compute方法中实现分治拆分后的计算。

Java实例5. SWT Layout

在图形框架中,支持多种类型的布局是基本功能。对Java来说,SWT就是一个很好的示例。

SWT FillLayoutSWT FillLayoutSWT RowLayoutSWT
RowLayout

不同的布局方式,需要提供一种方便且可扩展的设计。让我们来看看SWT中对不同Layout是如何设计的:

  • Context:SWT的抽象元素Composite实际执行的方法computerSize()。按照用户设定的布局计算大小进行绘制。
  • Strategy:SWT通过抽象类Layout提供策略接口。
  • ConcreteStrategy:具体的实现,比如FillLayout、RowLayout等,根据自身的计算方法计算布局。

总结

在策略模式上篇,介绍设计模式的基本结构、要解决的问题并列举了Java中策略模式使用的几个示例。下篇中,对具体的示例进行了分析,主要了解了:

  • 排序工具类Collections中sort的基本实现:通过比较器comparator实现不同的对象比较策略。
  • 正则表达式的Pattern实现:通过符合语法规则的正则表达式,以及匹配参数,实现不同的匹配策略。
  • 线程池ThreadPoolExecutor实现:通过提供不同的排队策略和拒绝策略,在线程池ThreadPoolExecutor的管理中,提供不同的管理策略。
  • ForkJoin框架的ForkJoinPool:通过提供并发任务线程池管理,对可分治解决的计算任务,开发者只需通过继承RecursiveAction或RecursiveTask即可实现不同的算法。
  • 图形框架SWT Layout:对于需要实现多种不同布局的图形元素,通过Layout接口即可满足各种需求的布局策略。

可以看到,策略模式中的Strategy在Java中既可以是interface,也可以是 abstract class,甚至可以是构造函数(比如正则表达式的Pattern)。但无论实现的方式如何,要解决的核心问题还是将业务实现与具体的算法(策略)分离。

本篇的示例只是Java框架、类、库中很少的一部分。目的希望在了解策略模式的同时,可以发现身边的策略模式实例,能够认出并说出为什么要采用策略模式。以此作为抛砖引玉,欢迎大家交流讨论。

【上篇】
【下篇】

抱歉!评论已关闭.