說(shuō)了那么多,相信你對(duì)小步快跑的概念有了一個(gè)初步的印象,但理解還不是很深。讓我們來(lái)看一看一個(gè)實(shí)際工作中的例子,來(lái)親身感受一下什么是大布局,什么是大設(shè)計(jì),什么是小設(shè)計(jì)。
還是回到前面那個(gè)Hello World的例子,起初的需求總是簡(jiǎn)單而清晰的。當(dāng)用戶登錄一個(gè)網(wǎng)站時(shí),網(wǎng)站往往需要給用戶打一個(gè)招呼:“hi, XXX! ”。同時(shí),如果此時(shí)是上午則顯示“Good morning! ”,如果是下午則顯示“Good afternoon! ”,除此顯示“Good night! ”。對(duì)于這樣一個(gè)需求我們?cè)谝粋€(gè)HelloWorld類中寫了十來(lái)行代碼:
如果需求沒(méi)有變更,一切都是美好的。但事情總是這樣,當(dāng)軟件第一次提交,變更就開(kāi)始了。系統(tǒng)總是不能直接獲得用戶名稱,而是先獲得他的userId,然后通過(guò)userId從數(shù)據(jù)庫(kù)中獲得用戶名。后面的問(wèn)候可能需要更加精細(xì),如中午問(wèn)候“Good noon! ”、傍晚問(wèn)候“Good evening! ”、午夜問(wèn)候“Good midnight! ”。除此之外,用戶希望在一些特殊的節(jié)日,如新年問(wèn)候“Happy new year! ”、情人節(jié)問(wèn)候“Happy valentine’s day! ”、三八婦女節(jié)問(wèn)候“Happy women’s day! ”,等等。除了已經(jīng)列出的節(jié)日,他們還希望臨時(shí)添加一些特殊的日子,因此問(wèn)候語(yǔ)需要形成一個(gè)庫(kù),并支持動(dòng)態(tài)添加。不僅如此,這個(gè)問(wèn)候庫(kù)應(yīng)當(dāng)支持多語(yǔ)言,如選擇英語(yǔ)則顯示“Good morning! ”,而選擇中文則顯示“上午好!”……總之,各種不同的需求被源源不斷地被用戶提出來(lái),因此我們的設(shè)計(jì)師開(kāi)始頭腦發(fā)熱、充血、開(kāi)始思維混亂。是的,如果你期望你自己能一步到位搞定所有這些需求,你必然會(huì)感到千頭萬(wàn)緒、顧此失彼,進(jìn)而做出錯(cuò)誤的設(shè)計(jì)。但如果你學(xué)會(huì)了“小步快跑”的開(kāi)發(fā)模式,一切就變得沒(méi)有那么復(fù)雜了。
首先,我們觀察原程序,發(fā)現(xiàn)它包含三個(gè)相對(duì)獨(dú)立的功能代碼段,因此我們采用重構(gòu)中的“抽取方法”,將它們分別抽取到三個(gè)函數(shù)getHour(), getFirstGreeting(), getSecondGreeting()中,并讓原函數(shù)對(duì)其引用:
這次重構(gòu)雖然使程序結(jié)構(gòu)發(fā)生了較大變化,但其中真正執(zhí)行的代碼卻沒(méi)有變化,還是那些代碼。隨后,我們核對(duì)需求發(fā)現(xiàn),用戶需求分成了兩個(gè)不同的分支:對(duì)用戶問(wèn)候語(yǔ)的變更,和關(guān)于時(shí)間的問(wèn)候語(yǔ)變更。為此,我們?cè)俅螌?duì)HelloWorld的程序進(jìn)行了分裂,運(yùn)用重構(gòu)中的“抽取類”,將對(duì)用戶問(wèn)候的程序分裂到GreetingToUser類中,將關(guān)于時(shí)間的問(wèn)候程序分裂到GreetingAboutTime類中:
系統(tǒng)重構(gòu)到這一步,我們來(lái)看看用戶關(guān)于時(shí)間問(wèn)候語(yǔ)部分的變更需求:?jiǎn)柡蛐枰泳?xì),如中午問(wèn)候“Good noon! ”、傍晚問(wèn)候“Good evening! ”、午夜問(wèn)候“Good midnight! ”。除此之外,用戶希望在一些特殊的節(jié)日,如新年問(wèn)候“Happy new year! ”、情人節(jié)問(wèn)候“Happy valentine’s day! ”、三八婦女節(jié)問(wèn)候“Happy women’s day! ”,等等。此時(shí)我們發(fā)現(xiàn),我們對(duì)時(shí)間問(wèn)候語(yǔ)的變更不再需要修改HelloWorld或其它什么類,而是僅僅專注于修改GreetingAboutTime就可以了,這就是因重構(gòu)帶來(lái)的改善。
同時(shí),我們發(fā)現(xiàn),過(guò)去只需getHour()就足夠,而現(xiàn)在卻需要getMonth()與getDay()。隨著程序復(fù)雜度的提升,我們適時(shí)進(jìn)行了一次重構(gòu),將與時(shí)間相關(guān)的程序抽取到一個(gè)新類DateUtil中,就可以順利地改寫原有的時(shí)間問(wèn)候語(yǔ)程序:
最后,我們建立user表存放用戶信息,創(chuàng)建UserDao接口及其實(shí)現(xiàn)類,為GreetingToUser提供用戶信息訪問(wèn)的服務(wù);我們用greetingRule表存放各種問(wèn)候語(yǔ),創(chuàng)建GreetingRuleDao接口及其實(shí)現(xiàn)類,為GreetingAboutTime提供一個(gè)可擴(kuò)展的、支持多語(yǔ)言的問(wèn)候語(yǔ)庫(kù)(如圖3.1所示)。所有這一切都是在現(xiàn)有基礎(chǔ)上,通過(guò)小步快跑的方式一步一步演變的。
小步快跑是一種逐步進(jìn)化式的程序優(yōu)化過(guò)程,它是重構(gòu)思想的重要核心。后面我們還會(huì)用更多實(shí)際工作中的示例,讓你真實(shí)體會(huì)到小步快跑的開(kāi)發(fā)過(guò)程。
大話重構(gòu)連載首頁(yè): http://fangang.iteye.com/blog/2081995
特別說(shuō)明:希望網(wǎng)友們?cè)谵D(zhuǎn)載本文時(shí),應(yīng)當(dāng)注明作者或出處,以示對(duì)作者的尊重,謝謝!
還是回到前面那個(gè)Hello World的例子,起初的需求總是簡(jiǎn)單而清晰的。當(dāng)用戶登錄一個(gè)網(wǎng)站時(shí),網(wǎng)站往往需要給用戶打一個(gè)招呼:“hi, XXX! ”。同時(shí),如果此時(shí)是上午則顯示“Good morning! ”,如果是下午則顯示“Good afternoon! ”,除此顯示“Good night! ”。對(duì)于這樣一個(gè)需求我們?cè)谝粋€(gè)HelloWorld類中寫了十來(lái)行代碼:
/** * The Refactoring's hello-world program * @author fangang */ public class HelloWorld { /** * Say hello to everyone * @param now * @param user * @return the words what to say */ public String sayHello(Date now, String user){ //Get current hour of day Calendar calendar = Calendar.getInstance(); calendar.setTime(now); int hour = calendar.get(Calendar.HOUR_OF_DAY); //Get the right words to say hello String words = null; if(hour>=6 && hour<12){ words = "Good morning!"; }else if(hour>=12 && hour<19){ words = "Good afternoon!"; }else{ words = "Good night!"; } words = "Hi, "+user+". "+words; return words; } }
如果需求沒(méi)有變更,一切都是美好的。但事情總是這樣,當(dāng)軟件第一次提交,變更就開(kāi)始了。系統(tǒng)總是不能直接獲得用戶名稱,而是先獲得他的userId,然后通過(guò)userId從數(shù)據(jù)庫(kù)中獲得用戶名。后面的問(wèn)候可能需要更加精細(xì),如中午問(wèn)候“Good noon! ”、傍晚問(wèn)候“Good evening! ”、午夜問(wèn)候“Good midnight! ”。除此之外,用戶希望在一些特殊的節(jié)日,如新年問(wèn)候“Happy new year! ”、情人節(jié)問(wèn)候“Happy valentine’s day! ”、三八婦女節(jié)問(wèn)候“Happy women’s day! ”,等等。除了已經(jīng)列出的節(jié)日,他們還希望臨時(shí)添加一些特殊的日子,因此問(wèn)候語(yǔ)需要形成一個(gè)庫(kù),并支持動(dòng)態(tài)添加。不僅如此,這個(gè)問(wèn)候庫(kù)應(yīng)當(dāng)支持多語(yǔ)言,如選擇英語(yǔ)則顯示“Good morning! ”,而選擇中文則顯示“上午好!”……總之,各種不同的需求被源源不斷地被用戶提出來(lái),因此我們的設(shè)計(jì)師開(kāi)始頭腦發(fā)熱、充血、開(kāi)始思維混亂。是的,如果你期望你自己能一步到位搞定所有這些需求,你必然會(huì)感到千頭萬(wàn)緒、顧此失彼,進(jìn)而做出錯(cuò)誤的設(shè)計(jì)。但如果你學(xué)會(huì)了“小步快跑”的開(kāi)發(fā)模式,一切就變得沒(méi)有那么復(fù)雜了。
首先,我們觀察原程序,發(fā)現(xiàn)它包含三個(gè)相對(duì)獨(dú)立的功能代碼段,因此我們采用重構(gòu)中的“抽取方法”,將它們分別抽取到三個(gè)函數(shù)getHour(), getFirstGreeting(), getSecondGreeting()中,并讓原函數(shù)對(duì)其引用:
/** * The Refactoring's hello-world program * @author fangang */ public class HelloWorld { /** * Say hello to everyone * @param now * @param user * @return the words what to say */ public String sayHello(Date now, String user){ int hour = getHour(now); return getFirstGreeting(user)+getSecondGreeting(hour); } /** * Get current hour of day. * @param now * @return current hour of day */ private int getHour(Date now){ Calendar calendar = Calendar.getInstance(); calendar.setTime(now); return calendar.get(Calendar.HOUR_OF_DAY); } /** * Get the first greeting. * @param user * @return the first greeting */ private String getFirstGreeting(String user){ return "Hi, "+user+". "; } /** * Get the second greeting. * @param hour * @return the second greeting */ private String getSecondGreeting(int hour){ if(hour>=6 && hour<12){ return "Good morning!"; }else if(hour>=12 && hour<19){ return "Good afternoon!"; }else{ return "Good night!"; } } }
這次重構(gòu)雖然使程序結(jié)構(gòu)發(fā)生了較大變化,但其中真正執(zhí)行的代碼卻沒(méi)有變化,還是那些代碼。隨后,我們核對(duì)需求發(fā)現(xiàn),用戶需求分成了兩個(gè)不同的分支:對(duì)用戶問(wèn)候語(yǔ)的變更,和關(guān)于時(shí)間的問(wèn)候語(yǔ)變更。為此,我們?cè)俅螌?duì)HelloWorld的程序進(jìn)行了分裂,運(yùn)用重構(gòu)中的“抽取類”,將對(duì)用戶問(wèn)候的程序分裂到GreetingToUser類中,將關(guān)于時(shí)間的問(wèn)候程序分裂到GreetingAboutTime類中:
/** * The Refactoring's hello-world program * @author fangang */ public class HelloWorld { /** * Say hello to everyone * @param now * @param user * @return the words what to say */ public String sayHello(Date now, String user){ GreetingToUser greetingToUser = new GreetingToUser(user); GreetingAboutTime greetingAboutTime = new GreetingAboutTime(now); return greetingToUser.getGreeting() + greetingAboutTime.getGreeting(); } } /** * The greeting to user * @author fangang */ public class GreetingToUser { private String user; /** * The constructor with user * @param user */ public GreetingToUser(String user){ this.user = user; } /** * @return greeting to user */ public String getGreeting(){ return "Hi, "+user+". "; } } /** * The greeting about time. * @author fangang */ public class GreetingAboutTime { private Date date; public GreetingAboutTime(Date date){ this.date = date; } /** * @param date * @return the hour of day */ private int getHour(Date date){ Calendar calendar = Calendar.getInstance(); calendar.setTime(date); return calendar.get(Calendar.HOUR_OF_DAY); } /** * @return the greeting about time */ public String getGreeting(){ int hour = getHour(date); if(hour>=6 && hour<12){ return "Good morning!"; }else if(hour>=12 && hour<19){ return "Good afternoon!"; }else{ return "Good night!"; } } }
系統(tǒng)重構(gòu)到這一步,我們來(lái)看看用戶關(guān)于時(shí)間問(wèn)候語(yǔ)部分的變更需求:?jiǎn)柡蛐枰泳?xì),如中午問(wèn)候“Good noon! ”、傍晚問(wèn)候“Good evening! ”、午夜問(wèn)候“Good midnight! ”。除此之外,用戶希望在一些特殊的節(jié)日,如新年問(wèn)候“Happy new year! ”、情人節(jié)問(wèn)候“Happy valentine’s day! ”、三八婦女節(jié)問(wèn)候“Happy women’s day! ”,等等。此時(shí)我們發(fā)現(xiàn),我們對(duì)時(shí)間問(wèn)候語(yǔ)的變更不再需要修改HelloWorld或其它什么類,而是僅僅專注于修改GreetingAboutTime就可以了,這就是因重構(gòu)帶來(lái)的改善。
同時(shí),我們發(fā)現(xiàn),過(guò)去只需getHour()就足夠,而現(xiàn)在卻需要getMonth()與getDay()。隨著程序復(fù)雜度的提升,我們適時(shí)進(jìn)行了一次重構(gòu),將與時(shí)間相關(guān)的程序抽取到一個(gè)新類DateUtil中,就可以順利地改寫原有的時(shí)間問(wèn)候語(yǔ)程序:
/** * The utility of time * @author fangang */ public class DateUtil { private Calendar calendar; /** * @param date */ public DateUtil(Date date){ calendar = Calendar.getInstance(); calendar.setTime(date); } /** * @return the hour of day */ public int getHour(){ return calendar.get(Calendar.HOUR_OF_DAY); } /** * @return the month of date */ public int getMonth(){ return calendar.get(Calendar.MONTH)+1; } /** * @return the day of month */ public int getDay(){ return calendar.get(Calendar.DAY_OF_MONTH); } } /** * The greeting about time. * @author fangang */ public class GreetingAboutTime { private Date date; public GreetingAboutTime(Date date){ this.date = date; } /** * @return the greeting about time */ public String getGreeting(){ DateUtil dateUtil = new DateUtil(date); int month = dateUtil.getMonth(); int day = dateUtil.getDay(); int hour = dateUtil.getHour(); if(month==1 && day==1) return "Happy new year! "; if(month==1 && day==14) return "Happy valentine's day! "; if(month==3 && day==8) return "Happy women's day! "; if(month==5 && day==1) return "Happy Labor day! "; ...... if(hour>=6 && hour<12) return "Good morning!"; if(hour==12) return "Good noon! "; if(hour>=12 && hour<19) return "Good afternoon! "; if(hour>=19 && hour<22) return "Good evening! "; return "Good night! "; } }
最后,我們建立user表存放用戶信息,創(chuàng)建UserDao接口及其實(shí)現(xiàn)類,為GreetingToUser提供用戶信息訪問(wèn)的服務(wù);我們用greetingRule表存放各種問(wèn)候語(yǔ),創(chuàng)建GreetingRuleDao接口及其實(shí)現(xiàn)類,為GreetingAboutTime提供一個(gè)可擴(kuò)展的、支持多語(yǔ)言的問(wèn)候語(yǔ)庫(kù)(如圖3.1所示)。所有這一切都是在現(xiàn)有基礎(chǔ)上,通過(guò)小步快跑的方式一步一步演變的。
圖3.1 HelloWorld的設(shè)計(jì)圖
小步快跑是一種逐步進(jìn)化式的程序優(yōu)化過(guò)程,它是重構(gòu)思想的重要核心。后面我們還會(huì)用更多實(shí)際工作中的示例,讓你真實(shí)體會(huì)到小步快跑的開(kāi)發(fā)過(guò)程。
大話重構(gòu)連載首頁(yè): http://fangang.iteye.com/blog/2081995
特別說(shuō)明:希望網(wǎng)友們?cè)谵D(zhuǎn)載本文時(shí),應(yīng)當(dāng)注明作者或出處,以示對(duì)作者的尊重,謝謝!
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號(hào)聯(lián)系: 360901061
您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對(duì)您有幫助就好】元
