国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 學(xué)院 > 開(kāi)發(fā)設(shè)計(jì) > 正文

從ifelse到設(shè)計(jì)模式的轉(zhuǎn)變

2019-11-18 11:15:23
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友
  面向過(guò)程設(shè)計(jì)和面向?qū)ο笤O(shè)計(jì)的主要區(qū)別是:是否在業(yè)務(wù)邏輯層使用冗長(zhǎng)的if else判定。假如你還在大量使用if else,當(dāng)然,界面表現(xiàn)層除外,即使你使用java/C#這樣完全面向?qū)ο蟮恼Z(yǔ)言,也只能說(shuō)明你的思維停留在傳統(tǒng)的面向過(guò)程語(yǔ)言上。

傳統(tǒng)思維習(xí)慣分析


  為什么會(huì)業(yè)務(wù)邏輯層使用if else,其實(shí)使用者的目的也是為了重用,但是這是面向過(guò)程編程的重用,程序員只看到代碼重用,因?yàn)樗吹絠f else幾種情況下大部分代碼都是重復(fù)的,只有個(gè)別不同,因此使用if else可以避免重復(fù)代碼,并且認(rèn)為這是模板Template模式。


  他范的錯(cuò)誤是:程序員只從代碼運(yùn)行順序這個(gè)方向來(lái)看待它的代碼,這種思維類似水管或串行電路,水沿著水管流動(dòng)(代碼運(yùn)行次序),當(dāng)碰到幾個(gè)分管(子管),就分到這幾個(gè)分管子在流動(dòng),這里就相當(dāng)于碰到代碼的if else處了。


  而使用OO,則首先打破這個(gè)代碼由上向下順序等同于運(yùn)行時(shí)的先后循序這個(gè)規(guī)律,代碼結(jié)構(gòu)不由執(zhí)行循序決定,由什么決定呢?由OO設(shè)計(jì);設(shè)計(jì)模式會(huì)取代這些if else,但是最后總是由一個(gè)Service等總類按照運(yùn)行順序組裝這些OO模塊,只有一處,這處可包含事務(wù),一般就是Service,EJB中是session bean。


  一旦需求變化,我們更多的可能是Service中各個(gè)OO模塊,甚至是只改動(dòng)Service中的OO模塊執(zhí)行順序就能符合需求。


  這里我們也看到OO分離的思路,將以前過(guò)程語(yǔ)言的一個(gè)Main函數(shù)徹底分解,將運(yùn)行順序與代碼其他邏輯分離開(kāi)來(lái),而不是象面向過(guò)程那樣混亂在一起。所以有人感慨,OO也是要順序的,這是肯定的,要害是運(yùn)行順序要單獨(dú)分離出來(lái)。


  是否有if else可以看出你有沒(méi)有將運(yùn)行順序分離到家。


設(shè)計(jì)模式的切入口


  經(jīng)常有人反映,設(shè)計(jì)模式是不錯(cuò),但是我很難用到,其實(shí)假如你使用if else來(lái)寫(xiě)代碼時(shí)(除顯示控制以外),就是在寫(xiě)業(yè)務(wù)邏輯,只不過(guò)使用簡(jiǎn)單的判定語(yǔ)句來(lái)作為現(xiàn)實(shí)情況的替代者。


   還是以大家熟悉的論壇帖子為例子,如ForumMessage是一個(gè)模型,但是實(shí)際中帖子分兩種性質(zhì):主題貼(第一個(gè)根貼)和回帖(回以前帖子的帖子),這里有一個(gè)樸素的解決方案:
建立一個(gè)ForumMessage,然后在ForumMessage加入isTopic這樣判定語(yǔ)句,注重,你這里一個(gè)簡(jiǎn)單屬性的判定引入,可能導(dǎo)致你的程序其他地方到處存在if else 的判定。


  假如我們改用另外一種分析實(shí)現(xiàn)思路,以對(duì)象化概念看待,實(shí)際中有主題貼和回帖,就是兩種對(duì)象,但是這兩種對(duì)象大部分是一致的,因此,我將ForumMessage設(shè)為表達(dá)主題貼;然后創(chuàng)建一個(gè)繼續(xù)ForumMessage的子類ForumMessageReply作為回帖,這樣,我在程序地方,如Service中,我已經(jīng)確定這個(gè)Model是回帖了,我就直接下溯為ForumMessageReply即可,這個(gè)有點(diǎn)類似向Collection放入對(duì)象和取出時(shí)的強(qiáng)制類型轉(zhuǎn)換。通過(guò)這個(gè)手段我消滅了以后程序中if else的判定語(yǔ)句出現(xiàn)可能。


  從這里體現(xiàn)了,假如分析方向錯(cuò)誤,也會(huì)導(dǎo)致誤用模式。


  討論設(shè)計(jì)模式舉例,不能沒(méi)有業(yè)務(wù)上下文場(chǎng)景的案例,否則無(wú)法決定是否該用模式,下面舉兩個(gè)對(duì)比的例子:


  第一. 這個(gè)帖子中舉例的第一個(gè)代碼案例是沒(méi)有上下文的,文中只說(shuō)明有一段代碼:





main() {

if(case A){


//do with strategy A


}else(case B){


//do with strategy B


}else(case C){


//do with strategy C


}


}


 

  這段代碼只是純粹的代碼,沒(méi)有業(yè)務(wù)功能,所以,在這種情況下,我們就很難確定使用什么模式,就是一定用策略模式等,也逃不過(guò)還是使用if else的命運(yùn),設(shè)計(jì)模式不是魔法,不能將一段毫無(wú)意義的代碼變得簡(jiǎn)單了,只能將其體現(xiàn)的業(yè)務(wù)功能更加輕易可拓展了。


  第二.在這個(gè)帖子中,作者舉了一個(gè)PacketParser業(yè)務(wù)案例,這段代碼是體現(xiàn)業(yè)務(wù)功能的,是一個(gè)數(shù)據(jù)包的分析,作者也比較了各種模式使用的不同,所以我們還是使用動(dòng)態(tài)代理模式或Command模式來(lái)消滅那些可能存在的if else


  由以上兩個(gè)案例表明:業(yè)務(wù)邏輯是我們使用設(shè)計(jì)模式的切入點(diǎn),而在分解業(yè)務(wù)邏輯時(shí),我們習(xí)慣則可能使用if else來(lái)實(shí)現(xiàn),當(dāng)你有這種企圖或者已經(jīng)實(shí)現(xiàn)代碼了,那么就應(yīng)該考慮是否需要重構(gòu)Refactoring了。


if else替代者


  那么實(shí)戰(zhàn)中,哪些設(shè)計(jì)模式可以替代if else呢?其實(shí)GoF設(shè)計(jì)模式都可以用來(lái)替代if else,我們分別描述如下:


  • 狀態(tài)模式 
      當(dāng)數(shù)據(jù)對(duì)象存在各種可能性的狀態(tài),而且這種狀態(tài)將會(huì)影響到不同業(yè)務(wù)結(jié)果時(shí),那么我們就應(yīng)該考慮是否使用狀態(tài)模式,當(dāng)然,使用狀態(tài)模式之前,你必須首先有內(nèi)存狀態(tài)這個(gè)概念,而不是數(shù)據(jù)庫(kù)概念,因?yàn)樵趥鹘y(tǒng)的面向過(guò)程的/面向數(shù)據(jù)庫(kù)的系統(tǒng)中,你很難發(fā)現(xiàn)狀態(tài)的,從數(shù)據(jù)庫(kù)中讀取某個(gè)值,然后根據(jù)這個(gè)值進(jìn)行代碼運(yùn)行分流,這是很多初學(xué)者常干的事情。參考文章:狀態(tài)對(duì)象:數(shù)據(jù)庫(kù)的替代者
      使用傳統(tǒng)語(yǔ)言思維的情況還有:使用一個(gè)類整數(shù)變量標(biāo)識(shí)狀態(tài):





     


    public class Order{




  • 發(fā)表評(píng)論 共有條評(píng)論
    用戶名: 密碼:
    驗(yàn)證碼: 匿名發(fā)表
    主站蜘蛛池模板: 海兴县| 微山县| 云霄县| 保定市| 来宾市| 浪卡子县| 马龙县| 沐川县| 永新县| 长海县| 化州市| 邯郸市| 荥阳市| 潼关县| 大厂| 海兴县| 精河县| 秀山| 保德县| 辛集市| 广元市| 始兴县| 秭归县| 铅山县| 屯留县| 雷山县| 万载县| 河间市| 通江县| 调兵山市| 大埔县| 浙江省| 长白| 将乐县| 兴安县| 平乡县| 元阳县| 张北县| 枝江市| 景东| 南部县|