前言 
上次v3版本,我們將Entity, Service, Dao, Utility都放到了類別庫裡面,讓我們可以輕鬆的在不同專案中用同一份組件。雖然文章沒有獲得太多的讚賞,不過相信那一定是太多人會這一招了。如果您已經會了,恭喜你,這是很重要的一步,沒有類別庫,後面我們很多事情都不容易實作出來。

今天要講的運用是interface,相信很多人都還是interface苦手,大部分的人還是卡在『為什麼我要用interface』,當我帶出可惡的PM需求時,大家應該會感同身受,而且覺得相當熟悉。跟著文章中的步伐前進,您將會知道,原來interface運用可以這麼簡單,這麼有用!

需求說明
越來越可惡的PM提出了另外一個需求:『上次您將商業邏輯跟資料存取放到了類別庫,讓我們的批次可以一起使用,這個idea實在太棒了!我們現在有另一個網站,也有個Validate的頁面,也想使用AuthenticationService,不過我們網站後面的資料庫都是Oracle的,資料結構也不一樣,那可以用同一份AuthenticationService嗎?』

當然可以!讀完這篇文章之後,希望您也可以這樣大聲的跟PM講:『當然可以!』。

先簡單列出,我們的功能需求:

  1. 頁面一樣
  2. 商業邏輯一樣
  3. DB來源不一樣

我們先來看「通常」大家拿到這一份需求,可能會怎麼做:

簡單嘛,我們多傳一個參數給AuthenticationService,來判斷是哪一個網站呼叫的,如果是Oracle的網站,就換呼叫Oracle的Dao方法。如果是原本的SQL server網站,就呼叫原本SQL的Dao方法。一個步驟就解決了,帥吧!

所以我們的程式就會變成這樣:

AuthenticationService: (很聰明的用了之前學到的手法) 

Validate.aspx.cs 

Console的Main() 

接著就會發現,原本用到AuthenticationService.VerifyPasswordById都要改,都要新增一個參數: site,這對我們來說很困擾,為什麼我為了一個新網站的需求,卻要『大幅』修改原本使用這個Service的程式。(您可能在很多地方都用到這個Service),完全違背了開放-封閉原則。或許您是使用VB.NET的,會說『簡單啊,我用optional來標示這個參數,那我就可以只為了新的Oracle網站來滿足新的需求即可。

為了這樣的需求,而採用了optional來標示參數,是一種慢性毒藥。會逐漸腐蝕您系統的架構到無法自拔。當optional參數個數超過4個的時候,您就會發現這個service方法的邏輯根本沒有可維護性。這樣的設計會導致內聚力太低,同樣的service甚至同一個方法裡面,包含了太多混雜的職責,所以隨便新增一個需求,就會讓程式動彈不得,越陷越深。

山不轉路轉,另一種常見的作法也是種毒藥,我們新增一個Service的方法,讓Oracle的Website呼叫不就得了?這樣之前的Code就不用改啦,又可以滿足新的需求。

程式如下: 

重構後: 

這樣不是很簡單明瞭嗎?

我們來看Oracle Website在使用的時候: 

在用這個類別庫的人,一定會有這個疑問,這兩個方法有什麼不同?這會讓職責混淆,使用上容易誤用。還有一個很嚴重的問題,萬一以後是從Excel檔案來呢?從Access來呢?從txt檔來呢?從其他web service來呢?越來越多的需求,我們的Architecture就會越蓋越歪,最後垮下來而無法彌補。

那,我們該怎麼解決這個問題?對,用Interface!!

設計步驟:
先把剛剛的code都砍掉(笑)! 我們重新思考一下,原本PM提出來的需求是,只有資料存取的部分不一樣,但『商業邏輯的部分完全一樣』,我們希望可以多一個資料存取功能是滿足新的需求。也就是給Oracle website用的仍然是同一個AuthenticationService的VerifyPasswordById的方法,這是不能變也不想變的。而對於Oracle的資料存取方法,也仍然需要傳入id,才能得到對應的password。

步驟一:
我們先在原本的AuthenticationDao的QueryPasswordById方法上,按滑鼠右鍵=>重構=>擷取介面。 

把我們的QueryPasswordById方法打勾,按下確定。 

接著我們原本的AuthenticationDao後面就會多出來 : IAuthenticationDao 

而產生的介面也相當簡單: 

步驟二:
新增一個AuthenticationDaoForOracle的類別在DataAccess的folder底下,實作IAuthenticationDao: 

會看到Visual Studio自動幫我們產生了要實作(遵守)介面的方法: 

接著我們就可以不理它了!(笑)

步驟三:
接著來調整我們的AuthenticationService,很簡單地!我們將原本public的MyAuthenticationDao的Property,將型別從AuthenticationDao改成IAuthenticationDao。讓service原本直接呼叫AuthenticationDao的相依性,轉成相依於IAuthenticationDao這個介面上,而不直接相依於某一個特定類別。 

這個時候,其實我們的方案,建置是會成功的。我們的所有邏輯也都撰寫完畢了,是的,就這麼簡單。我們已經滿足了,service用同一份,Dao資料來源不同的設計了,接著,我們只是要做組合的動作。

步驟四:
回到我們原本有用到AuthenticationService的程式中,我們要多做一件事:將我們要用的Dao(也就是concrete class),塞給AuthenticationService。請各位想像一下,當我們在步驟三,將AuthenticationService開了一個介面出來給外面,就像一塊拼圖開一個特定的凹洞出來。有實作這個介面的class,就能滿足這個凹洞,他們就可以組合在一起,發揮不同的功能。

接著,我們來設定一下中斷點,看一下程式是否跟原本一樣,是使用AuthenticationDao來存取資料: 

大家想像自己的程式,就像以前的聖戰士,或是百獸王,我們的程式,就是一個一個的元件,用的人可以任意的組合他們,只要能夠『插』(injection)的起來。

最後,我們也將Oracle website的程式修改一下。我們希望在Oracle的website,使用AuthenticationService的時候,後面是接著AuthenticationDaoForOracle這個元件的。 

當執行偵錯,就會看到最後是進入AuthenticationDaoForOracle的中斷點: 

最後我們的程式架構如下圖所示,正規來說,Service也應該要有對應的interface,讓頁面可以只相依於Service的Interface,讓Service也可以抽換。最後就達成我們3-layer: Presentation layer (頁面、UI), Business logic layer(Service class), Persistence layer(Data access object),都有透過interface來隔絕layer與layer之間的相依性,讓我們的系統架構可以有彈性的抽換,以及無限的擴充性,也可以滿足開放-封閉原則。 

步驟三補充說明:
步驟三中,我們提到將原本的public propery型別直接改成介面,並交給外面來set。這『可能』會導致一個問題,就是當外界使用AuthenticationService,卻沒有assign MyAuthenticationDao的時候,會出現NullReferenceException。就像使用的人沒有告知AuthenticationService後面要使用的元件,導致方法走到後面就斷掉了。

雖然會出現這樣的潛在問題,但這樣設計是很合理的使用狀況。如果真的要限制,不能出現這類的狀況,也就是強迫使用這個Service的人,一定要assign MyAuthenticationDao,我們可以在AuthenticationService的建構式,加入IAuthenticationDao的參數,讓使用AuthenticationService的場景,在new的時候,一定要給IAuthenticationDao的concrete class。 

有人或許會說,如果在建構式中assign了IAuthenticationDao的concrete class的instance,那MyAuthenticationDao這個屬性是不是就可以乾脆開成private,基本上,是!

這兩種作法,哪一種都可以,但大家可以想像,如果我這個Service用到很多外部類別,那麼我的建構式不就超長一串?是的,而且這是合理的情況。為了節省每次要用,都要new一堆concrete class的instance塞給我們要用的service,所以會有DI framework的出現。(DI=Dependency injection),透過DI framework,我們可以把『組合』這件事,寫的更輕鬆,而且獨立出來統一管理,不會散落一地。而DI framework,有的有支援auto-wiring,也就是framework在碰到建構式有需要的型別時,會自動填入你設定好的concrete class的instance。有的有支援injection public property。所以,採用哪一種寫法,其實可以因應不同的DI framework來設計,基本上兩種都OK啦。

結論
透過上面的需求跟實際操作,相信大家已經知道為什麼我們要使用interface,以及使用上的概念就像組裝一樣。這也是為什麼interface通常會被解釋成『合約』的概念,因為實作了這個合約,這個class就要做出凸出來的那一塊,只要有人有一樣凹的情況,就要能拿這個凸的class去接。

    1. 使用了interface,其實間接的就是實作了IoC的概念。原本我們的Authentication.Verify(),裡面用到QueryPasswordById()是相依於AuthenticationDao上。透過介面,我們的AuthenticationService是相依於IAuthenticatoinDao介面上。這就是IoC(控制反轉)的概念。
      這樣的設計,我們相依的這個介面,就像一個凹口,後面可以有很多很多種凸出來的class來接,這樣我們在使用時就可以任意組裝。 
    2. 使用Interface可以讓關注點分離,讓設計的邏輯穩定。我們的系統結構變成下圖所示: 

    3. 除了讓原本的邏輯可以穩定不變以外,透過Interface,更為未來的無限擴充奠下了穩固的架構: 
    4. 增加可測試性。(這個就留到後面的重構再來談囉…)

转:[ASP.NET]重構之路系列v4 – 簡單使用interface之『你也會IoC』的更多相关文章

  1. [ASP.NET MVC 小牛之路]04 - 依赖注入(DI)和Ninject

    本人博客已转移至:http://www.exblr.com/liam  为什么需要依赖注入 在[ASP.NET MVC 小牛之路]系列的理解MVC模式文章中,我们提到MVC的一个重要特征是关注点分离( ...

  2. [ASP.NET MVC 小牛之路]05 - 使用 Ninject

    在[ASP.NET MVC 小牛之路]系列上一篇文章(依赖注入(DI)和Ninject)的末尾提到了在ASP.NET MVC中使用Ninject要做的两件事情,续这篇文章之后,本文将用一个实际的示例来 ...

  3. [ASP.NET MVC 小牛之路]06 - 使用 Entity Framework

    在家闲着也是闲着,继续写我的[ASP.NET MVC 小牛之路]系列吧.在该系列的上一篇博文中,在显示书本信息列表的时候,我们是在程序代码中手工造的数据.本文将演示如何在ASP.NET MVC中使用E ...

  4. [ASP.NET MVC 大牛之路]01 - 开篇

    匆匆2014,转眼就到末尾了.在这一年,你还有哪事情些想做而没有做? 2014年在我身上发生了两件意义重大的事,一是正月初一宝宝出生,我升级成为了爸爸:二是进入了一家创业公司,成为了技术负责人. 去年 ...

  5. [ASP.NET MVC 大牛之路]02 - C#高级知识点概要(1) - 委托和事件

    在ASP.NET MVC 小牛之路系列中,前面用了一篇文章提了一下C#的一些知识点.照此,ASP.NET MVC 大牛之路系列也先给大家普及一下C#.NET中的高级知识点.每个知识点不太会过于详细,但 ...

  6. [ASP.NET MVC 小牛之路]05 - 使用 Ninject实现依赖注入

    在[ASP.NET MVC 小牛之路]系列上一篇文章(依赖注入(DI)和Ninject)的末尾提到了在ASP.NET MVC中使用Ninject要做的两件事情,续这篇文章之后,本文将用一个实际的示例来 ...

  7. [ASP.NET MVC 小牛之路]10 - Controller 和 Action (2)

    继上一篇文章之后,本文将介绍 Controller 和 Action 的一些较高级特性,包括 Controller Factory.Action Invoker 和异步 Controller 等内容. ...

  8. [ASP.NET MVC 小牛之路]13 - Helper Method

    我们平时编程写一些辅助类的时候习惯用“XxxHelper”来命名.同样,在 MVC 中用于生成 Html 元素的辅助类是 System.Web.Mvc 命名空间下的 HtmlHelper,习惯上我们把 ...

  9. [ASP.NET MVC 小牛之路]15 - Model Binding

    Model Binding(模型绑定)是 MVC 框架根据 HTTP 请求数据创建 .NET 对象的一个过程.我们之前所有示例中传递给 Action 方法参数的对象都是在 Model Binding ...

随机推荐

  1. 【随笔】android开发的学习路线

    第一阶段:Java面向对象编程 1.Java基本数据类型与表达式,分支循环. 2.String和StringBuffer的使用.正则表达式. 3.面向对象的抽象,封装,继承,多态,类与对象,对象初始化 ...

  2. BZOJ1444 : [Jsoi2009]有趣的游戏

    建立AC自动机,并求出转移矩阵. 再用$\sum E(终止节点)=1$去替换第一个方程,高斯消元即可. 时间复杂度$O(n^3l^3)$. 注意精度问题,要特判0.00的情况. #include< ...

  3. HDU 2612 (BFS搜索+多终点)

    题目链接: http://poj.org/problem?id=1947 题目大意:两人选择图中一个kfc约会.问两人到达时间之和的最小值. 解题思路: 对于一个KFC,两人的BFS目标必须一致. 于 ...

  4. HDU 1180 (BFS搜索)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1180 题目大意:迷宫中有一堆楼梯,楼梯横竖变化.这些楼梯在奇数时间会变成相反状态,通过楼梯会顺便到达 ...

  5. svg―Raphael.js Library

    Raphael是一个用于在网页中绘制矢量图形的Javascript库,它使用SVG W3C推荐标准和VML作为创建图形的基础,可以通过JavaScript操作DOM来轻松创建出各种复杂的柱状图.饼图. ...

  6. LIS+LCS+LCIS

    PS:本篇博文均采用宏#define FOR(i, a, n) for(i = a; i <= n; ++i) LIS:最长上升子序列 废话不多说:http://baike.baidu.com/ ...

  7. 深度解析开发项目之 05 - 解决textField编辑之后点击其他内容改变的问题

    深度解析开发项目之 05 - 解决textField编辑之后点击其他内容改变的问题 问题的解决:  只需要给HeadeVIew加上这句代码

  8. sql对应C#的类型

  9. itellyou MSDN, 我告诉你 win7系统工具等

    http://www.itellyou.cn/ 操作系统-window7-中文-Windows 7 Ultimate with Service Pack 1 (x64) - DVD (Chinese- ...

  10. 对于PKI(公钥基础结构)及证书服务的通俗理解

    对于PKI及证书服务的这些概念,相信初学者会有许多迷惑的地方,那是因为其中的某些关键概念没有理解清楚,我力争以通俗易懂的方式给初学者一些启示,也给以后自己忘了的时候一个参考:) ! 参考资料:http ...