日期:2012-04-07  浏览次数:20392 次

工厂方法

THE FACTORY METHOD

我们已经学会了简单工厂模式(Simple Factory Pattern),工厂(factory)思想贯穿于整个面向对象编程(OOP)以及其他一些设计模式的始终。如:生成器(Builder)模式。其间,一个单一的类扮演类似交通警察的角色,决定哪一个单一层次上的子类将被实例化。

工厂方法模式(Factory Method pattern)是对工厂(factory)思想进行了巧妙的延伸,它使得将超类的实例化延迟到它的每一个子类。这个模式没有具体的指出延迟到哪一个子类,而是定义一个抽象类创建对象,让其子类决定创建哪一个对象。

下面是一个简单的例子,在一个游泳比赛中如何确定游泳运动员的泳道。按照运动员的成绩排列决赛事的分组,速度越快所分的小组的决赛的次序越靠后,反之,速度越慢就越先比赛,而且在每一个小组中成绩越好、速度越快的选手也就越靠近中间的泳道。这被称为straight seeding。

当游泳运动员在锦标赛比赛过程中,他们通常要游两次。 通过在预赛中相互竞争,前12名或者16名游泳运动员将继续在决赛中继续彼此竞争。 为了预赛工作更公平, 采用circle seeded,这样使得速度最快的3名选手分别处于最迅速3个小组的中心泳道。在剩下的选手中再选出速度最好的3名选手,等等。

我们要实现这个选拔模式并且使用工厂方法。

首先,设计抽象事件类:

Public Class Events

Protected numLanes As Integer

Protected swmmers As Swimmers

'-----

Public Sub New(ByVal Filename As String, ByVal lanes As Integer)

MyBase.New()

Dim s As String

Dim sw As Swimmer

Dim fl As vbFile

fl = New vbFile(Filename) '打开一个文本文件

fl.OpenForRead()

numLanes = lanes '保存泳道数量信息

swmmers = New Swimmers

'读取游泳选手信息

s = fl.readLine

While Not fl.fEOF

sw = New Swimmer(s) '建立对象

swmmers.Add(sw) 'add to list

s = fl.readLine

End While

fl.closeFile()

End Sub

'-----

Public Function getSwimmers() As ArrayList

getSwimmers = swmmers

End Function

'-----

Public Overridable Function isPrelim() As Boolean

End Function

'-----

Public Overridable Function isFinal() As Boolean

End Function

'-----

Public Overridable Function isTimedFinal() As Boolean

End Function

'-----

Public Overridable Function getSeeding() As Seeding

End Function



End Class



因为所有的派生类都要从文本文件读取数据,所以,我们把Events类作为基类。其中所定义的方法均为虚方法,可以通过继承Events类来实现具体的类(PrelimEvent类、TimedFinalEvent类),这两个类之间唯一的不同就是返回的选拔的类别不同。我们也定义了一个包含以下方法的抽象选拔类:



Public MustInherit Class Seeding

Protected numLanes As Integer

Protected laneOrder As ArrayList

Protected numHeats As Integer

Private asw() As Swimmer

Protected sw As Swimmers

'-----

Public Function getSeeding() As Swimmers

getSeeding = sw

End Function

'-----

Public Function getHeat() As Integer

End Function

'-----

Public Function getCount() As Integer

getCount = sw.Count

End Function

'-----

Public MustOverride Sub seed()

'-----

Public Function getSwimmers() As ArrayList

getSwimmers = sw

End Function

'-----

Public Function getHeats() As Integer

Return numHeats

End Function

'-----

Public Function odd(ByVal n As Integer) As Boolean

odd = (n \ 2) * 2 <> n

End Function

'-----

Public Function calcLaneOrder(ByVal lns As Integer) As ArrayList

numLanes = lns

Dim lanes(numLanes) As Integer

Dim i As Integer

Dim mid, incr, ln As Integer

mid = (numLanes \ 2)

If (odd(numLanes)) Then

mid = mid + 1

End If

incr = 1

ln = mid

For i = 0 To numLanes - 1

lanes(i) = ln