<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆-6  評論-55  文章-0  trackbacks-0
    面向過程設計和面向對象設計的主要區別是:是否在業務邏輯層使用冗長的if else判斷。如果你還在大量使用if else,當然,界面表現層除外,即使你使用Java/C#這樣完全面向對象的語言,也只能說明你的思維停留在傳統的面向過程語言上。

    傳統思維習慣分析

      為什么會業務邏輯層使用if else,其實使用者的目的也是為了重用,但是這是面向過程編程的重用,程序員只看到代碼重用,因為他看到if else幾種情況下大部分代碼都是重復的,只有個別不同,因此使用if else可以避免重復代碼,并且認為這是模板Template模式。

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

      而使用OO,則首先打破這個代碼由上向下順序等同于運行時的先后循序這個規律,代碼結構不由執行循序決定,由什么決定呢?由OO設計;設計模式會取代這些if else,但是最后總是由一個Service等總類按照運行順序組裝這些OO模塊,只有一處,這處可包含事務,一般就是Service,EJB中是Session bean。

      一旦需求變化,我們更多的可能是Service中各個OO模塊,甚至是只改動Service中的OO模塊執行順序就能符合需求。

      這里我們也看到OO分離的思路,將以前過程語言的一個Main函數徹底分解,將運行順序與代碼其他邏輯分離開來,而不是象面向過程那樣混亂在一起。所以有人感慨,OO也是要順序的,這是肯定的,關鍵是運行順序要單獨分離出來。

      是否有if else可以看出你有沒有將運行順序分離到家。

    設計模式的切入口

      經常有人反映,設計模式是不錯,但是我很難用到,其實如果你使用if else來寫代碼時(除顯示控制以外),就是在寫業務邏輯,只不過使用簡單的判斷語句來作為現實情況的替代者。

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

      如果我們改用另外一種分析實現思路,以對象化概念看待,實際中有主題貼和回帖,就是兩種對象,但是這兩種對象大部分是一致的,因此,我將ForumMessage設為表達主題貼;然后創建一個繼承ForumMessage的子類ForumMessageReply作為回帖,這樣,我在程序地方,如Service中,我已經確定這個Model是回帖了,我就直接下溯為ForumMessageReply即可,這個有點類似向Collection放入對象和取出時的強制類型轉換。通過這個手段我消滅了以后程序中if else的判斷語句出現可能。

      從這里體現了,如果分析方向錯誤,也會導致誤用模式。

      討論設計模式舉例,不能沒有業務上下文場景的案例,否則無法決定是否該用模式,下面舉兩個對比的例子:

      第一. 這個帖子中舉例的第一個代碼案例是沒有上下文的,文中只說明有一段代碼:

    main() { 
    if(case A){ 

    //do with strategy A 

    }else(case B){

    //do with strategy B 

    }else(case C){ 

    //do with strategy C 

    }


       

      這段代碼只是純粹的代碼,沒有業務功能,所以,在這種情況下,我們就很難確定使用什么模式,就是一定用策略模式等,也逃不過還是使用if else的命運,設計模式不是魔法,不能將一段毫無意義的代碼變得簡單了,只能將其體現的業務功能更加容易可拓展了。

      第二.在這個帖子中,作者舉了一個PacketParser業務案例,這段代碼是體現業務功能的,是一個數據包的分析,作者也比較了各種模式使用的不同,所以我們還是使用動態代理模式或Command模式來消滅那些可能存在的if else

      由以上兩個案例表明:業務邏輯是我們使用設計模式的切入點,而在分解業務邏輯時,我們習慣則可能使用if else來實現,當你有這種企圖或者已經實現代碼了,那么就應該考慮是否需要重構Refactoring了。


    if else替代者

      那么實戰中,哪些設計模式可以替代if else呢?其實GoF設計模式都可以用來替代if else,我們分別描述如下:

    狀態模式 
      當數據對象存在各種可能性的狀態,而且這種狀態將會影響到不同業務結果時,那么我們就應該考慮是否使用狀態模式,當然,使用狀態模式之前,你必須首先有內存狀態這個概念,而不是數據庫概念,因為在傳統的面向過程的/面向數據庫的系統中,你很難發現狀態的,從數據庫中讀取某個值,然后根據這個值進行代碼運行分流,這是很多初學者常干的事情。參考文章:狀態對象:數據庫的替代者
      使用傳統語言思維的情況還有:使用一個類整數變量標識狀態:
     

    public class Order{

    private int status;

    //說明: 

    //status=1 表示訂貨但為查看 ;

    //status=2 表示已經查看未處理;

    //status=3 表示已經處理未付款

    //status=4 表示已經付款未發貨

    //status=5 表示已經發貨


     

      上述類設計,無疑是將類作為傳統語言的函數來使用,這樣導致程序代碼中存在大量的if else。


    策略模式 
      當你面臨幾種算法或者公式選擇時,可以考慮策略模式,傳統過程語言情況是:從數據庫中讀取算法數值,數值1表示策略1,例如保存到數據庫;數值為2表示策略2,例如保存到XMl文件中。這里使用if else作為策略選擇的開關。 


    command模式 
      傳統過程的思維情況是:如果客戶端發出代號是1或"A",那么我調用A.java這個對象來處理;如果代號是2或"B",我就調用B.java來處理,通過if else來判斷客戶端發送過來的代碼,然后按事先約定的對應表,調用相應的類來處理。


    MVC模式 
      MVC模式的傳統語言誤用和Command模式類似,在一個Action類中,使用if else進行前后臺調度,如果客戶端傳送什么命令;我就調用后臺什么結果;如果后臺處理什么結構,再決定推什么頁面,不過,現在我們使用Struts/JSF這樣MVC模式的框架實現者就不必范這種低級錯誤。


    職責鏈模式 
      職責鏈模式和Command模式是可選的,如果你實在不知道客戶端會發出什么代號;也沒有一個事先定義好的對照表,那么你只能編寫一個個類去碰運氣一樣打開這個包看一下就可以。與Command是不同在AOP vs Decorator一文中有分析。


    代理或動態代理模式 
      代理對象可以是符合某種條件的代表者,比如,權限檢驗,傳統面向過程思維是:當一個用戶登陸后,訪問某資源時,使用if else進行判斷,只有某種條件符合時,才能允許訪問,這樣權限判斷和業務數據邏輯混亂在一起,使用代理模式可以清晰分離,如果嫌不太好,使用動態代理,或者下面AOP等方式。


    AOP或Decorator模式
      
      其實使用filter過濾器也可以替代我們業務中的if else,過濾器起到一種過濾和篩選作用,將符合本過濾器條件的對象攔截下來做某件事情,這就是一個過濾器的功能,多個過濾器組合在一起實際就是if else的組合。
      所以,如果你實在想不出什么辦法,可以使用過濾器,將過濾器看成防火墻就比較好理解,當客戶端有一個請求時,經過不同性質的防火墻,這個防火墻是攔截端口的;那個防火墻是安全檢查攔截等等。過濾器也如同紅藍白各種光濾鏡;紅色濾鏡只能將通過光線中的紅色攔截了;藍色濾鏡將光線中的藍色攔截下來,這實際上是對光線使用if else進行分解。


      如圖,通過一個個條件過濾器我們立體地實現了對信號的分離,如果你使用if else,說明你是將圖中的條件1/2/3/4合并在一起,在同一個地方實現條件判斷。
      需要深入了解過濾器的實現細節和微小區別,請參考文章:AOP vs Decorator

    OO設計的總結   

      還有一種偽模式,雖然使用了狀態等模式,但是在模式內部實質還是使用if else或switch進行狀態切換或重要條件判斷,那么無疑說明還需要進一步努力。更重要的是,不能以模式自居,而且出書示人。

      真正掌握面向對象這些思想是一件困難的事情,目前有各種屬于揪著自己頭發向上拔的解說,都是誤人子弟的,所以我覺得初學者讀Thinking in Java(Java編程思想)是沒有用,它試圖從語言層次來講OO編程思想,非常失敗,作為語言參考書可以,但是作為Java體現的OO思想的學習資料,就錯了。

      OO編程思想是一種方****,方****如果沒有應用比較,是無法體會這個方****的特點的,禪是古代一個方****,悟禪是靠挑水砍柴這些應用才能體會。

      那么OO思想靠什么應用能夠體會到了?是GoF設計模式,GoF設計模式是等于軟件人員的挑水砍柴等基本活,所以,如果一個程序員連基本活都不會,他何以自居OO程序員?從事OO專業設計編程這個工作,如果不掌握設計模式基本功,就象一個做和尚的人不愿意挑水砍柴,他何以立足這個行業?早就被師傅趕下山。

      最后總結:將if else用在小地方還可以,如簡單的數值判斷;但是如果按照你的傳統習慣思維,在實現業務功能時也使用if else,那么說明你的思維可能需要重塑,你的編程經驗越豐富,傳統過程思維模式就容易根深蒂固,想靠自己改變很困難;建議接受專業頭腦風暴培訓。

      用一句話總結:如果你做了不少系統,很久沒有使用if else了,那么說明你可能真正進入OO設計的境地了。(這是本人自己發明的實戰性的衡量考核標準)。
    posted on 2007-11-11 19:53 心酸果凍 閱讀(2722) 評論(13)  編輯  收藏 所屬分類: JAVAEE

    評論:
    # re: 你還在用if else嗎? 2007-11-11 20:13 | 驚鴻逝水
    這文章怎么看得眼熟?好像是Jdon網站上Copy的?  回復  更多評論
      
    # re: 你還在用if else嗎? 2007-11-11 21:43 | sitinspring
    同志,轉載還是要寫出處的,實在不知道也應該加上轉貼二字.  回復  更多評論
      
    # re: 你還在用if else嗎?(轉) 2007-11-11 22:52 | sunday
    哦 問問你一個問題 如果有個業務對象有15個狀態 我在action類中用if else判斷其狀態 做不同的操作 那現在是否要寫15個對象來完成問題
    在頁面中展示 每個狀態用不同的圖片表示 不用if else 是否要寫15個頁面來表示  回復  更多評論
      
    # re: 你還在用if else嗎?(轉) 2007-11-12 08:58 | BeanSoft
    哎....人心不古

    不過面向對象的Java底層是面向過程的C實現的, 呵呵 一樣一樣的啊  回復  更多評論
      
    # re: 你還在用if else嗎?(轉)[未登錄] 2007-11-12 11:09 | paul
    有時if 。else是不可避免的,如果使用設計模式來解決的話,也許會從擴展性等方面帶來很多好處,但你肯定會增加很多的小類,有時也會對性能有影響,所以什么時候用,怎么用設計模式是個關鍵,別濫用設計模式  回復  更多評論
      
    # re: 你還在用if else嗎?(轉)[未登錄] 2007-11-12 12:26 | 曲強 Nicky
    實際上If then在很多規則邏輯的狀況下是不可避免的,而且即使是上面文章中所說的oo設計也不可能完全保證代碼的重用性,這個時候規則如何來表達就是要研究的話題,更重要的是當規則易變的情況下。在這種情況下,我們通常使用Rule Engine 比如clips jess等等其他if then的表現引擎來驅動多變得規則。  回復  更多評論
      
    # re: 你還在用if else嗎?(轉) 2007-11-12 12:39 | 心酸果凍
    @曲強 Nicky
    聽了各位的發言,感覺似乎都有道理,但是我們要根據實際的情況去,解決問題,找到最好的辦法.  回復  更多評論
      
    # re: 你還在用if else嗎?(轉) 2007-11-12 19:04 | sitinspring
    @曲強 Nicky

    能否撰文介紹clips jess等.
      回復  更多評論
      
    # re: 你還在用if else嗎?(轉)[未登錄] 2007-11-13 13:40 | 曲強 Nicky
    @sitinspring
    等等會的,我正在做相關的一個項目和我的課畢設,也在實現自己的一個RE  回復  更多評論
      
    # re: 你還在用if else嗎?(轉)[未登錄] 2007-11-14 09:12 | kingofhawks
    何必如此痛恨if else....  回復  更多評論
      
    # re: 你還在用if else嗎?(轉) 2007-11-14 09:27 | zkdemon
    不錯的文章!!!  回復  更多評論
      
    # re: 你還在用if else嗎?(轉) 2007-11-18 08:40 | sitinspring
    @曲強 Nicky

    靜候佳音.  回復  更多評論
      
    # re: 你還在用if else嗎?(轉)[未登錄] 2011-07-04 10:33 | hj
    @sunday
    自己回答的很好啊  回復  更多評論
      
    主站蜘蛛池模板: 国产91免费在线观看| 青柠影视在线观看免费高清| 国产精彩免费视频| 久久精品国产亚洲av麻豆色欲 | 美女的胸又黄又www网站免费| 一二三四在线播放免费观看中文版视频 | 成人免费网站视频www| 情侣视频精品免费的国产| 亚洲一卡2卡三卡4卡无卡下载| 免费在线看v网址| 亚洲国产成人资源在线软件 | 国产免费av片在线无码免费看| 精品亚洲国产成人av| 免费乱码中文字幕网站| 国产va免费精品| 亚洲AV无码成人网站久久精品大 | 永久看日本大片免费35分钟| 亚洲高清资源在线观看| 中文字幕影片免费在线观看| 亚洲一区二区三区写真| 国产一区二区三区无码免费| 亚洲精品视频免费观看| 久久国产亚洲电影天堂| 国产精品色拉拉免费看| 亚洲AV无码一区二区三区牲色| 又大又硬又爽免费视频| 一个人看的www免费视频在线观看 一个人免费视频观看在线www | 精品无码AV无码免费专区| 国产成人精品日本亚洲网址| 国产无遮挡吃胸膜奶免费看 | 98精品全国免费观看视频| 亚洲高清一区二区三区| 亚洲精品无码久久毛片| 久久一区二区三区免费播放| 中文字幕无码精品亚洲资源网久久 | 可以免费观看的国产视频| 亚洲一区二区三区久久| 亚洲精品无码专区2| 久久国产乱子免费精品| 亚洲AV无码一区二区三区牛牛| 亚洲av无码乱码在线观看野外|