日期:2013-11-08  浏览次数:20445 次

工厂模式

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)方法。这就是我们所说的隐藏了具体类。