小菜編程成長記(一 面試受挫——代碼無錯就是好?)第3/3頁
五 體會簡單工廠模式的美妙
次日,小菜再來找大鳥,問道:“你昨天說計算器這樣的小程序還可以用到面向?qū)ο笕筇匦??繼承和多態(tài)怎么可能用得上,我實(shí)在不可理解?!?
大鳥:“小菜很有鉆研精神嗎?好,今天我讓你功力加深一級。你先要考慮一下,你昨天寫的這個代碼,能否做到很靈活的可修改和擴(kuò)展呢?”
小菜:“我已經(jīng)把業(yè)務(wù)和界面分離了呀,這不是很靈活了嗎?”
大鳥:“那我問你,現(xiàn)在如果我希望增加一個開根(sqrt)運(yùn)算,你如何改?”
小菜:“那只需要改Operation類就行了,在switch中加一個分支就行了?!?
大鳥:“問題是你要加一個平方根運(yùn)算,卻需要把加減乘除的運(yùn)算都得來參與編譯,如果你一不小心,把加法運(yùn)算改成了減法,這不是大大的糟糕。打個比方,如果現(xiàn)在公司要求你為公司的薪資管理系統(tǒng)做維護(hù),原來只有技術(shù)人員(月薪),市場銷售人員(底薪+提成),經(jīng)理(年薪+股份)三種運(yùn)算算法,現(xiàn)在要增加兼職工作人員的(時薪)算法,但按照你昨天的程序?qū)懛?,公司就必須要把包含有的原三種算法的運(yùn)算類給你,讓你修改,你如果心中小算盤一打,‘TMD,公司給我的工資這么低,我真是郁悶,這會有機(jī)會了',于是你除了增加了兼職算法以外,在技術(shù)人員(月薪)算法中寫了一句
if (員工是小菜)
{
salary = salary * 1.1;
}
那就意味著,你的月薪每月都會增加10%(小心被抓去坐牢),本來是讓你加一個功能,卻使得原有的運(yùn)行良好的功能代碼產(chǎn)生了變化,這個風(fēng)險太大了。你明白了嗎?”
小菜:“哦,你的意思是,我應(yīng)該把加減乘除等運(yùn)算分離,修改其中一個不影響另外的幾個,增加運(yùn)算算法也不影響其它代碼,是這樣嗎?”
大鳥:“自己想去吧,如何用繼承和多態(tài),你應(yīng)該有感覺了。”
小菜:“OK,我馬上去寫。”
/// <summary>
/// 運(yùn)算類
/// </summary>
class Operation
{
private double _numberA = 0;
private double _numberB = 0;
/// <summary>
/// 數(shù)字A
/// </summary>
public double NumberA
{
get{ return _numberA; }
set{ _numberA = value;}
}
/// <summary>
/// 數(shù)字B
/// </summary>
public double NumberB
{
get{ return _numberB; }
set{ _numberB = value; }
}
/// <summary>
/// 得到運(yùn)算結(jié)果
/// </summary>
/// <returns></returns>
public virtual double GetResult()
{
double result = 0;
return result;
}
}
/// <summary>
/// 加法類
/// </summary>
class OperationAdd : Operation
{
public override double GetResult()
{
double result = 0;
result = NumberA + NumberB;
return result;
}
}
/// <summary>
/// 減法類
/// </summary>
class OperationSub : Operation
{
public override double GetResult()
{
double result = 0;
result = NumberA - NumberB;
return result;
}
}
/// <summary>
/// 乘法類
/// </summary>
class OperationMul : Operation
{
public override double GetResult()
{
double result = 0;
result = NumberA * NumberB;
return result;
}
}
/// <summary>
/// 除法類
/// </summary>
class OperationDiv : Operation
{
public override double GetResult()
{
double result = 0;
if (NumberB==0)
throw new Exception("除數(shù)不能為0。");
result = NumberA / NumberB;
return result;
}
}
小菜:“大鳥哥,我按照你說的方法寫出來了一部分,首先是一個運(yùn)算類,它有兩個Number屬性,主要用于計算器的前后數(shù),然后有一個虛方法GetResult(),用于得到結(jié)果,然后我把加減乘除都寫成了運(yùn)算類的子類,繼承它后,重寫了GetResult()方法,這樣如果要修改任何一個算法,都不需要提供其它算法的代碼了。但問題來了,我如何讓計算器知道我是希望用哪一個算法呢?”
大鳥:“寫得很不錯嗎,大大超出我的想象了,你現(xiàn)在的問題其實(shí)就是如何去實(shí)例化對象的問題,哈,今天心情不錯,再教你一招‘簡單工廠模式',也就是說,到底要實(shí)例化誰,將來會不會增加實(shí)例化的對象(比如增加開根運(yùn)算),這是很容易變化的地方,應(yīng)該考慮用一個單獨(dú)的類來做這個創(chuàng)造實(shí)例的過程,這就是工廠,來,我們看看這個類如何寫。”
/// <summary>
/// 運(yùn)算類工廠
/// </summary>
class OperationFactory
{
public static Operation createOperate(string operate)
{
Operation oper = null;
switch (operate)
{
case "+":
{
oper = new OperationAdd();
break;
}
case "-":
{
oper = new OperationSub();
break;
}
case "*":
{
oper = new OperationMul();
break;
}
case "/":
{
oper = new OperationDiv();
break;
}
}
return oper;
}
}
大鳥:“哈,看到吧,這樣子,你只需要輸入運(yùn)算符號,工廠就實(shí)例化出合適的對象,通過多態(tài),返回父類的方式實(shí)現(xiàn)了計算器的結(jié)果?!?BR>
Operation oper;
oper = OperationFactory.createOperate("+");
oper.NumberA = 1;
oper.NumberB = 2;
double result = oper.GetResult();
大鳥: “哈,界面的實(shí)現(xiàn)就是這樣的代碼,不管你是控制臺程序,Windows程序,Web程序,PDA或手機(jī)程序,都可以用這段代碼來實(shí)現(xiàn)計算器的功能,當(dāng)有一天我們需要更改加法運(yùn)算,我們只需要改哪里?”
小菜:“改OperationAdd 就可以了。”
大鳥: “那么我們需要增加各種復(fù)雜運(yùn)算,比如平方根,立方根,自然對數(shù),正弦余弦等,如何做?”
小菜:“只要增加相應(yīng)的運(yùn)算子類就可以了呀?!?
大鳥: “嗯?夠了嗎?”
小菜:“對了,還需要去修改運(yùn)算類工廠,在switch中增加分支。”
大鳥: “哈,那才對,那如果要修改界面呢?”
小菜:“那就去改界面呀,關(guān)運(yùn)算什么事呀?!?
小菜:“ 回想那天我面試題寫的代碼,我終于明白我為什么寫得不成功了,原來一個小小的計算器也可以寫出這么精彩的代碼,謝謝大鳥?!?
大鳥: “吼吼,記住哦,編程是一門技術(shù),更加是一門藝術(shù),不能只滿足于寫完代碼運(yùn)行結(jié)果正確就完事,時??紤]如何讓代碼更加簡煉,更加容易維護(hù),容易擴(kuò)展和復(fù)用,只有這樣才可以是真的提高。寫出優(yōu)雅的代碼真的是一種很爽的事情。不過學(xué)無止境,其實(shí)這才是理解面向?qū)ο蟮拈_始呢。給你出個作業(yè),做一個商場收銀軟件,營業(yè)員根據(jù)客戶購買商品單價和數(shù)量,向客戶收費(fèi)?!?
小菜:“就這個?沒問題呀。”
相關(guān)文章
C# DataTable 轉(zhuǎn)換為 實(shí)體類對象實(shí)例
如果你的實(shí)體類與數(shù)據(jù)庫表是完全一致的。上代碼:2013-04-04解析C#設(shè)計模式編程中適配器模式的實(shí)現(xiàn)
這篇文章主要介紹了C#設(shè)計模式編程中適配器模式的實(shí)現(xiàn),分別舉了類的對象適配器與對象的適配器模式的例子,需要的朋友可以參考下2016-02-02提權(quán)函數(shù)之RtlAdjustPrivilege()使用說明
RtlAdjustPrivilege() 這玩意是在 NTDLL.DLL 里的一個不為人知的函數(shù),MS沒有公開,原因就是這玩意實(shí)在是太NB了,以至于不需要任何其他函數(shù)的幫助,僅憑這一個函數(shù)就可以獲得進(jìn)程ACL的任意權(quán)限!2011-06-06C#使用Dns類實(shí)現(xiàn)查詢主機(jī)名對應(yīng)IP地址
C#中的Dns類能夠與默認(rèn)的DNS服務(wù)器進(jìn)行通信,以檢索IP地址,這篇文章主要介紹了C#如何使用Dns類解析出主機(jī)對應(yīng)的IP地址信息,需要的可以參考下2024-02-02C#隨機(jī)設(shè)置900-1100毫秒延遲的方法
這篇文章主要介紹了C#隨機(jī)設(shè)置900-1100毫秒延遲的方法,涉及C#中Thread.Sleep方法的使用技巧,需要的朋友可以參考下2015-04-04深入分析C#中WinForm控件之Dock順序調(diào)整的詳解
本篇文章是對C#中WinForm控件之Dock順序調(diào)整進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05C#中DataSet、DataTable、DataRow數(shù)據(jù)的復(fù)制方法
這篇文章介紹了C#中DataSet、DataTable、DataRow數(shù)據(jù)的復(fù)制方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-07-07結(jié)合.net框架在C#派生類中觸發(fā)基類事件及實(shí)現(xiàn)接口事件
這篇文章主要介紹了結(jié)合.net框架在C#派生類中觸發(fā)基類事件及實(shí)現(xiàn)接口事件,示例的事件編程中包括接口和類的繼承等面向?qū)ο蟮幕A(chǔ)知識,需要的朋友可以參考下2016-02-02