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