對抽象工廠+反射+配置檔案的例項理解
第一遍學習設計模式時,總有飄飄然的感覺.《大話設計模式》這本書雖然讀起來挺有趣味,也感覺比其他講設計模式的書好理解一些。但是真要用在機房收費系統時還是費勁的。米老師講過:思想要上移,行動要下移。這兩點要緊密結合,缺一不可。總在思想層面上繞,比如設計模式中所舉的生活中的例子,用在程式碼中卻不是那麼容易,所以必須要行動下移,認真實踐。而總在程式碼中繞,不進行總結提煉概括,學習就很侷限,所以我們也要不斷總結,思想上移。第二遍學習抽象工廠是在機房收費系統中,我想講講使用它的過程。
一.為什麼要用抽象工廠
以備我們的軟體要在不同資料庫上執行。現在我們使用的是SQL Sever,而要使用Access或者Oracle等時就需要考慮了。抽象工廠可以使我們靈活地更換到其他資料庫上。如何能夠靈活結合UML圖進行理解。
圖中的紅色框是IDAL(介面)。我舉個生活中的例子接電的插口來理解。同一個插口我們可以插電腦,電視,電冰箱等等。只要插頭與插口的規格一致就可以。對於這張包中可以看到整個邏輯從UI到IDAL就結束了,後面的D層和Helper要看具體情況,實現介面的可以使SqlDAL,也可以使AccessDAL。
二.反射和配置檔案是起什麼作用的。
《大話設計模式》中講過抽象工廠的優點和缺點。缺點之一就是需求增加時,比如要增加一張表。就要至少更改三個類。IFactory,SqlFactory,AccessFacotry...所以採用了簡單工廠來改進抽象工廠,而在簡單工廠中的例項化需要通過分支判斷語句來判斷。這裡就引入了反射,反射主要來解決分支判斷的問題,使得例項化更加地方便。具體如何靈活需要通過例項講解。
反射使用格式:Assembly.Load("程式集名稱").CreateInstance("名稱空間.類名稱“)
剛開始感覺很暈,不明白程式集名稱應該是什麼?名稱空間又是指得哪個?反射是用來解決分支判斷問題的,使得例項化更加方便。程式集一定是要具體例項化的地方。這裡就是指D層,預設情況,程式集名稱和名稱空間是同樣的。這裡都是SqlDAL,當然如果是AccessDAL,程式集名稱就是指AccessDAL了。說它靈活是指可以使用配置檔案指定具體是Sql還是Access。而具體的類名稱就要看你實現的是什麼功能了。如果是登入,那麼就是UserDAL(類名稱)
三登入例項應用
我按一條線的順序給出程式碼:
B層:
''' <summary>
''' 驗證使用者是否存在
''' </summary>
''' <param name="enUser">使用者實體</param>
''' <returns>DataTable</returns>
''' <remarks></remarks>
Public Function ExistUser(ByVal enUser As Entity.UsersEntity) As DataTable
Dim iu As IDAL.IUser
Dim dt As New DataTable
iu = Factory.DataAccess.CreateUser() '由於是shared定義的函式,這裡直接呼叫,不用例項化。
dt = iu.SelectUser(enUser)
Return dt
End Function
工廠:
Private Shared ReadOnly db As String = ConfigurationManager.AppSettings("DB")
Private Shared ReadOnly assemblyName = db + "DAL"
''' <summary>
''' 使用反射和配置檔案建立相應的介面
''' </summary>
''' <returns>IUser介面</returns>
''' <remarks></remarks>
Public Shared Function CreateUser() As IDAL.IUser
Dim className As String = assemblyName + "." + db + "UserDAL"
Return CType(Assembly.Load(assemblyName).CreateInstance(className), IDAL.IUser) '呼叫反射
End Function
介面:
''' <summary>
''' 登入時查詢使用者是否存在.
''' </summary>
''' <param name="enUser">引數為使用者實體</param>
''' <returns>datatable ,將userPassword,level,作為表中的兩列返回</returns>
''' <remarks></remarks>
Function SelectUser(ByVal enUser As Entity.UsersEntity) As DataTable
D層的名稱空間是SqlDAL,類名稱為SqlUserDAL。
配置檔案:
<appSettings >
<add key="DB" value="Sql"/>
</appSettings>
四.總結:多實踐,多總結。