工厂模式
the factory pattern
在面向对象(oo)程序中,我们经常看到的一种模式就是简单工厂模式(simple factory pattern)。简单工厂模式根据提供给它的数据,返回几个可能类中的一个类的实例。通常它返回的类都有一个共同的父类和共同的方法,但每个方法执行的认为不同,而且根据不同的数据进行了优化。简单工厂模式实际上不属于23个gof模式,但它可以作为工厂方法模式(factory method)的一个引导。
工厂模式如何工作
为了便于理解简单工厂模式,可以看下图:
工厂类(creator)角色:该角色是工厂方法模式的核心,含有按照一定商业逻辑创建产品。工厂类在客户端的直接调用下创建产品对象,它往往由一个具体类实现。
抽象产品(product)角色:担任这个角色的类是工厂方法模式所创建的对象的父类,或它们共同拥有的接口。抽象产品角色可以用一个接口或者抽象类实现。
具体产品(concrete product)角色:工厂方法模式所创建的任何对象都是这个角色的实例,具体产品角色由一个具体类实现。
在上图中,类x为一个基类,类xy和xz派生自类x, xfactory类根据你提供给它的值来决定返回那个子类的实例,我们定义了getclass方法,接受一些值,并根据这个值返回x类的其中一个实例。返回它们当中的哪一个与程序员无关。因为它们都含有相同的方法,但是不同的实现,它可能是很复杂的功能,但通常都很简单。
类
现在,我们写两个简单的类,用于实现将一个字符串分割为两部分,假设我们有一个输入窗体,允许用户输入他的姓名,可我们通过简单工厂模式namefactory类来自动判断用户输入的firstname和lastname之间到底是空格还是逗号。如果是逗号或空格我们就将其分割为两部分。
先定义一个简单的类。
public class nameclass
protected lname, frname as string
public function getfirst() as string
return frname
end function
public function getlast() as string
return lname
end function
end class
下面编写两个派生类:
firstfirst类、lastfirst类
它们实现了基类nameclass,并在构造函数中将名字分成了两部分。
firstfirst类
public class firstfirst
inherits nameclass
public sub new(byval nm as string)
dim i as integer
i = nm.indexof(" ")
if i > 0 then
frname = nm.substring(0, i).trim()
lname = nm.substring(i + 1).trim()
else
frname = ""
lname = nm
end if
end sub
end class
lastfirst类
public class lastfirst
inherits nameclass
public sub new(byval nm as string)
dim i as integer
i = nm.indexof(",")
if i > 0 then
lname = nm.substring(0, i).trim()
frname = nm.substring(i + 1).trim()
else
frname = ""
lname = nm
end if
end sub
end class
构造简单工厂
现在很容易给出简单工厂类。只检测逗号是否存在,然后返回其中的一个类的实例。
public class namefactory
public function getnamer( _
byval nm as string) as nameclass
dim i as integer
i = nm.indexof(",")
if i > 0 then
return new lastfirst(nm)
else
return new firstfirst(nm)
end if
end function
end class
使用简单工厂
我们具体实现如下:
"执行"按钮的 click事件:
dim nm as string
dim i as integer
nm = txname.text
dim nmer as nameclass
nmer = new namefactory().getnamer(nm)
txlast.text = nmer.getlast()
txfirst.text = nmer.getfirst()
我们已经构造了一个简单的用户接口,并且允许用户输入名字。运行如图:
用户输入名字然后点击”执行”按钮,这个名字被namefactory接收,并根据不同的类型(由逗号、空格分隔),返回不同子类的实例。这里不需要知道使用的是哪一个派生类的实例,工厂(namefactory)为我们提供了这个类,所需要知道的就是它有两个(getfirst、getlast)方法。这就是我们所说的隐藏了具体类。