成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

代碼診所第一次診斷

開發(fā) 開發(fā)工具
我之前的一個團隊開發(fā)人員的現(xiàn)狀是:沒有Clean Code的意識,不知道何謂TDD與重構(gòu),寫出來的Java代碼質(zhì)量很糟。如果從項目開初不針對這一問題進行有效的防治,就可能導(dǎo)致整個代碼庫陷入泥沼之中。于是我當(dāng)起了診治代碼疾病的醫(yī)生。

[[186193]]

幾年前,我有機會負責(zé)一個項目的咨詢。團隊很小,目標是對舊有系統(tǒng)的后端用Java改寫,而團隊的開發(fā)人員全為C程序員。我的工作職責(zé)是負責(zé)項目設(shè)計、開發(fā),以及擔(dān)任項目開發(fā)過程敏捷化的教練,并培養(yǎng)Java開發(fā)人員。

C程序員的一個特點是基本不具備面向?qū)ο笾R,在初步掌握Java語法之后,寫出來的代碼還是過程化的代碼。團隊開發(fā)人員的現(xiàn)狀就是:沒有Clean Code的意識,不知道何謂TDD與重構(gòu),寫出來的Java代碼質(zhì)量很糟。如果從項目開初不針對這一問題進行有效的防治,就可能導(dǎo)致整個代碼庫陷入泥沼之中。

為此,我要求在每日站會之后及時開展了代碼評審活動。評審過程中,只能以我為主導(dǎo),幫助大家發(fā)現(xiàn)代碼的壞味道。從一開始,具有壞味道的代碼可謂俯拾皆是,就像我們每天都在呼吸污染了的空氣一般不可避免。要凈化空氣任重而道遠,要讓團隊成員寫出好的代碼,同樣任重而道遠。

得有藥方才行。

于是我當(dāng)起了診治代碼疾病的醫(yī)生。為了更容易傳播醫(yī)療知識,我在團隊工作室的墻角落,開了一個小小的診所,廣而告之——“每日一貼,包治百病”。剛開張時,診所門面還沒裝修好,所以直接找了個白板開出了一個藥方:

診治代碼疾病

個人認為,這些處方不僅僅對于當(dāng)時的客戶團隊有療效,可能也適合大多數(shù)開發(fā)團隊。幾年過去了,我把這些處方分享出來,也算是一個小小的總結(jié)吧。先來看看這些處方吧。

***條:應(yīng)隨時保持架構(gòu)的清晰與簡單:統(tǒng)一所有查詢?yōu)镽epository。

項目其實并不需要訪問數(shù)據(jù)庫,而是通過遠程的Telnet(或其他協(xié)議)去訪問前端的設(shè)備。然而,我們可以借鑒DDD中資源庫的這個隱喻。至于提到的架構(gòu),則是我在架構(gòu)設(shè)計時參考了DDD的分層邏輯架構(gòu)。

DDD的分層邏輯架構(gòu)

為保證架構(gòu)的簡單與清晰,我做了一些“一刀切”的簡化原則:例如對Repository和Service的定義。通過Telnet等網(wǎng)絡(luò)協(xié)議獲取設(shè)備信息的功能不妨看做是對DB的查詢。因而,諸如NodeConfigureGetter這樣的類就應(yīng)該統(tǒng)一命名為NodeConfigureRepository。

該處方的主要目的是為了保持代碼的一致性,若不加以規(guī)范,就會出現(xiàn)Getter、Finder、Query等各種不統(tǒng)一的類名后綴,讓人眼花繚亂。

第二條:依賴注入(對象之間的協(xié)作)

很多OO初學(xué)者并不能理解依賴注入。我的一個辦法是讓他們從可測試性的角度出發(fā)。例如,倘若在NodeConfigureRepository類中直接實例化了TelnetService(這個類提供連接、登錄、執(zhí)行命令等與Telnet有關(guān)的操作),那么該怎樣在不需要Telnet環(huán)境的基礎(chǔ)上為NodeConfigureRepository編寫單元測試呢?解決不了這樣的問題,就說明設(shè)計的可測試性不夠好。

解決方案就是依賴注入。當(dāng)時的項目并未引入第三方IoC容器,原因在于項目的Jar包需要和另一個系統(tǒng)協(xié)作,并駐留在Flash中。容量有限,不允許引入太多第三方包,保證Jar包的精悍。

第三條:方法名體現(xiàn)意圖。

這個問題是許多開發(fā)人員都容易犯的毛病,尤其對于面向過程設(shè)計的程序員而言,很少會站在對象的角度去思考方法(即行為,準確地說,從設(shè)計的角度講應(yīng)該是對象承擔(dān)的職責(zé))。例如在NodeConfigureRepository類中,開發(fā)人員定義了getNodeConfigure方法,但返回值卻是void:

  1. public NodeConfigureRepository { 
  2.     private NodeConfigure configure; 
  3.     public NodeConfigureRepository(NodeConfigure configure) { 
  4.         this.configure = configure; 
  5.     } 
  6.     public void getNodeConfigure() { 
  7.         getMasterLogicBoard(); 
  8.         getMasterIp(); 
  9.         getEnvId(); 
  10.         getMasterSlot(); 
  11.         getSlaveIp(); 
  12.         getSlaveBoardTypeAndStatus(); 
  13.     } 

這個方法調(diào)用的諸多私有方法實則都是對構(gòu)造函數(shù)傳入的NodeConfigure進行數(shù)據(jù)收集。這樣的定義不僅讓代碼的調(diào)用者感覺怪怪的,測試也變得極為詭異:

  1. @Test 
  2. public void should_get_main_ctrl_logic_board_type() { 
  3.     configure = Nodeconfigure(); 
  4.     configureRepository = new NodeConfigureRepository(configure); 
  5.     configureRepository.getNodeConfigure(); 
  6.  
  7.     assertThat(configure.getMasterLogicBoard(), is(12288)); 

怎么改?

方法就是讓getNodeConfigure()方法直接返回組裝之后的NodeConfigure對象,并且解除NodeConfigure與NodeConfigureRepository之間的生命周期依賴。有趣的是getNodeConfigure方法內(nèi)調(diào)用的私有方法。它成了一種設(shè)計的例外,因為在Java中通常需要避免直接對輸入?yún)?shù)進行修改,并將其作為返回結(jié)果。而在這里出現(xiàn)的一系列方法,實則是履行對NodeConfigure對象的數(shù)據(jù)收集,因而可以定義為:

  1. private void  collectMasterLogicBoard(NodeConfigure configure) {} 
  2. private void  collectMasterIp(NodeConfigure configure) {} 
  3. private void  collectEnvId(NodeConfigure configure) {} 
  4. private void  collectMasterSlot(NodeConfigure configure) {} 
  5. private void  collectSlaveIp(NodeConfigure configure) {} 
  6. private void  collectSlaveBoardTypeAndStatus(NodeConfigure configure) {} 

我對這些方法的名稱進行了修改,使其能夠更好地展現(xiàn)其意圖。于是,getNodeConfigure()就變成了:

  1. public NodeConfigure getNodeConfigure() { 
  2.    NodeConfigure configure = new NodeConfigure(); 
  3.    collectMasterLogicBoard(configure); 
  4.    collectMasterIp(configure); 
  5.    collectEnvId(configure); 
  6.    collectMasterSlot(configure); 
  7.    collectSlaveIp(configure); 
  8.    collectSlaveBoardTypeAndStatus(configure); 
  9.    return configure;} 

這里實際上是Kent Beck提出的Collected Parameter模式。它是Visitor模式的簡化設(shè)計。當(dāng)然,我們也可以運用Builder模式對NodeConfigure對象進行組裝。

第四條:同一個方法中的實現(xiàn)代碼應(yīng)處于同一抽象層次。

這其實是老生常談了。Kent Beck在Smalltalk Best Practice Patterns一書中提到了“組合方法”模式,建議“讓一個方法中的所有操作處于相同的抽象層”,即所謂的SLAP原則。在Robert Martin的Clean Code一書中也反復(fù)提到這一原則,Neal Ford在Emergent Design也有詳細描述。

第五條:避免“啞對象”

這里展現(xiàn)的壞味道,在Martin Fowler的Refactoring一書中已有提及。在項目中,存在一些操作Xml文件的操作,并將這些Xml文件的Element映射為了Java對象。我們沒有使用Jaxb,因為對于我們有限的xml操作而言,Jaxb還是顯得太重。然而,在我們的代碼中,包括PackageStatusFileParser、StoragePackageGenerator、DownloadingConfigureParser等類中都存在著將Xml Element轉(zhuǎn)換為PackageInfo、SoftInfo等對象的重復(fù)代碼。

原因就在于我們將這些對象看做了“啞”的數(shù)據(jù)對象,而沒有將這種轉(zhuǎn)換行為封裝到擁有這些數(shù)據(jù)的對象中(我們的轉(zhuǎn)換僅牽涉到Xml,沒有擴展可能,因而無需使用Visitor模式)。

除了會導(dǎo)致大量的重復(fù)代碼之外,一旦轉(zhuǎn)換邏輯發(fā)生變化,例如XmlElement增加了Attribute,就可能需要到處修改,形成所謂的“霰彈式修改”。因而需要將這些邏輯封裝到對象中,例如:

  1. public class PackgeInfo { 
  2.     public PackageInfo createFrom(Elment element) {} 
  3. PackageInfo packageInfo = PackageInfo.createFrom(element); 

代碼雖為細節(jié)末道,看似微末,然而在項目開始之初不加以規(guī)范與約束,而任其腐化蔓延,最終造成的苦果還是會反擊到整個系統(tǒng);待到代碼質(zhì)量已經(jīng)墮落到不可修復(fù)的地步時,再要挽回,可能已經(jīng)覆水難收了。

【本文為51CTO專欄作者“張逸”原創(chuàng)稿件,轉(zhuǎn)載請聯(lián)系原作者】

戳這里,看該作者更多好文

責(zé)任編輯:趙寧寧 來源: 51CTO專欄
相關(guān)推薦

2011-07-21 21:01:37

諾基亞塞班蘋果

2017-03-24 21:26:26

代碼架構(gòu)Java

2023-09-11 00:14:46

后端團隊項目

2022-03-16 14:59:28

打包debian模板文件

2012-04-13 10:11:58

Windows 8泄露

2022-08-15 08:16:56

shiroWeb認證

2022-06-21 09:26:28

開源項目PR

2015-10-26 16:38:17

2021-02-05 08:35:21

私活程序員

2018-09-11 17:05:12

戴爾

2013-05-13 11:35:53

獨立開發(fā)開發(fā)經(jīng)驗開發(fā)感悟

2013-06-03 09:28:49

游戲設(shè)計

2010-05-27 10:00:09

2022-05-06 11:27:23

虛擬人白皮書行業(yè)

2024-07-09 10:20:05

VueJSX函數(shù)

2024-01-03 21:50:32

緩存機制請求

2012-01-18 11:18:12

Web App

2013-02-25 09:43:22

LambdasJava8

2018-11-21 14:51:00

Windows 功能系統(tǒng)

2018-08-15 10:34:30

戴爾
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 亚洲国产成人精品女人久久久 | 欧美激情国产日韩精品一区18 | 精品免费国产视频 | 国产精品久久久久久亚洲调教 | 一区在线视频 | 一级大黄色片 | 国产精品综合一区二区 | 红桃视频一区二区三区免费 | 久久久一区二区 | 日日夜夜免费精品视频 | 7799精品视频天天看 | 国产精品欧美一区二区 | 国产剧情一区二区三区 | 婷婷精品| 亚洲成人午夜电影 | 日日摸夜夜爽人人添av | 精品久久久久久久久久久下田 | 成人国产精品一级毛片视频毛片 | 免费99精品国产自在在线 | 亚洲欧美日韩在线一区二区 | 亚洲免费在线播放 | 国产成人精品午夜 | 99精品久久久 | 成人在线免费视频 | 日韩成人精品在线 | 在线中文字幕视频 | 国产精品国产精品国产专区不卡 | 中文字幕在线免费 | 欧美成人性生活 | 热久久性| av网站在线播放 | 成人片免费看 | 国产高清视频在线 | 亚洲在线免费观看 | 色黄网站 | 一区二区三区中文字幕 | 国产精品高潮呻吟久久av黑人 | 嫩草视频在线看 | 国产美女视频黄a视频免费 国产精品福利视频 | 国产精品久久国产精品 | av一级一片 |