设计模式之工厂方法(FACTORY METHOD))(一)
2024-07-21 02:23:37
供稿:网友
工厂方法
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
ln = mid + incr
incr = -incr
if (incr > 0) then
incr = incr + 1
end if
next i
laneorder = new arraylist
for i = 0 to numlanes - 1
laneorder.add(lanes(i))
next i
calclaneorder = laneorder
end function
public sub new(byval swmrs as swimmers, byval lanes as integer)
mybase.new()
sw = swmrs
numlanes = lanes
end sub
'-------------------
public function sort(byval sw as swimmers) as swimmers
dim i, j, max as integer
dim tmp as swimmer
try
max = sw.count
dim asw(max) as swimmer
for i = 0 to max - 1
asw(i) = ctype(sw.item(i), swimmer)
next i
for i = 0 to max - 1
for j = i to max - 1
if asw(i).gettime > asw(j).gettime then
tmp = asw(i)
asw(j) = asw(i)
asw(i) = tmp
end if
next j
next i
sw = new swimmers
for i = 0 to max - 1
sw.add(asw(i))
next i
sort = sw
catch e as exception
console.writeline("caught " + i.tostring + " " + j.tostring + " " + max.tostring + " " + e.tostring())
console.writeline(e.stacktrace)
end try
end function
end class