(續(xù)上期)
“空泛的討論讓人厭煩。”,Solmyr 笑容可掬的說道,“不如我們設(shè)定一個(gè)簡單的場景來看看你的計(jì)數(shù)器怎么使用吧。假設(shè)你是暴雪的程序員,要為星際爭霸設(shè)計(jì)程序表示神族的單位,那么最簡單的方案是 ——”,Solmyr 停了下來,望向 zero 。
zero 松了一口氣 —— 這個(gè)問題還不算困難。他在腦中整理了一下思路:“神族的單位應(yīng)該設(shè)計(jì)為一個(gè)基類,然后每種特定的兵種從這個(gè)類派生,每個(gè)單位就是這樣一個(gè)類的對象。”想到這里,他飛快的在百板上寫下:
class ProtossUnit
{
……
};
class Probe : public ProtossUnit
{
……
};
class Zealot : public ProtossUnit
{
……
};
class Dragoon : public ProtossUnit
{
……
};
Solmyr 點(diǎn)了點(diǎn)頭,接著說到:“很好。接下來,我們都知道星際爭霸里每個(gè)單位都是要占用人口的,也就是說我們得確切知道單位個(gè)數(shù),很明顯,這是一個(gè)對象計(jì)數(shù)的應(yīng)用。那么我們該怎樣利用你剛才實(shí)現(xiàn)的計(jì)數(shù)器呢?”
zero 順手就在白板上寫下:
class Probe : public ProtossUnit
{
……
Counter<Probe> m_MyCounter;
};
class Zeolot : public ProtossUnit
{
……
Counter<Zeolot> m_MyCounter;
};
class Dragoon : public ProtossUnit
{
……
Counter<Dragoon> m_MyCounter;
};
不對!!
zero 心中劃過警兆:這感覺太熟悉了!幾乎每次慘遭 Solmyr 毒手之前,都有這種感覺!他幾乎都可以感受到 Solmyr 正在尋找順手的東西來砸他。一定有什么地方不對了!
回過頭來看看自己寫下的東西,zero 很快的發(fā)現(xiàn)了自己的錯(cuò)誤:Counter<Zeolot> 和 Counter<Dragoon> 是不同的類,它們的計(jì)數(shù)值各自獨(dú)立,而星際爭霸中各兵種占用人口是共享的。
“既然是共享的,那么應(yīng)該加到基類里。”,zero 急急忙忙的擦去了上面兩行代碼,寫下:
class ProtossUnit
{
……
Counter<ProtossUnit> m_MyCounter;
};
還 …… 還是不對!zero 立刻又發(fā)現(xiàn)了問題:不同的兵種可能占用的人口數(shù)并不相同,象 Probe 就只占用一個(gè)人口,而 Zealot 和 Dragoon 就要占用兩個(gè),這 …… 這 ……
zero 再度擦去了剛寫下的代碼,站在白板之前舉棋不定。這時(shí) Solmyr 的聲音響了起來:“怎么了?有困難嗎?”。此時(shí) Solmyr 臉上的笑容顯得特別可惡。
“不,我只是不清楚星際爭霸中的人口是怎樣定義的,這個(gè)游戲我從來沒有玩過。”,zero 試圖拖延一點(diǎn)時(shí)間。
“是嗎?昨天我怎么還聽到你在討論‘星際爭霸神族戰(zhàn)術(shù)’?而且剛才你一下子就寫出了三個(gè)神族兵種的名稱,拼寫準(zhǔn)確。”,Solmyr 輕易的戳破了 zero 的謊言。
“……”,zero 不由得懊惱起來。“怎么辦?得讓它們共享一個(gè)計(jì)數(shù)器,而且每種兵種的計(jì)數(shù)值必須不一樣 …… 對了!”zero 腦中靈光一閃,寫下如下代碼:
class Probe : public ProtossUnit
{
……
Counter<ProtossUnit> m_MyCounter;
};
class Zeolot : public ProtossUnit
{
……
Counter<ProtossUnit> m_MyCounter;
Counter<ProtossUnit> m_MyCounter;
};
class Dragoon : public ProtossUnit
{
……
Counter<ProtossUnit> m_MyCounter;
Counter<ProtossUnit> m_MyCounter;
};
“Yeah!OK 了!”,zero 高興的喊道,全然不顧臺下帶著笑意的目光 —— 這樣的有趣場面已經(jīng)成了公司里的著名娛樂之一。“共享一個(gè)計(jì)數(shù)器的關(guān)鍵是用哪個(gè)類別作為模板參數(shù)!不一定非得把本身作為模板參數(shù),完全可以用各個(gè)兵種共同的基類!”
“那計(jì)數(shù)值不是 1 呢?”
“多放幾個(gè)計(jì)數(shù)器就行了!”
“嗯,還算不錯(cuò)。”
zero 很高興的看到 Solmyr 上前來在白板上打了一個(gè)勾,然而喜悅僅僅維持了一瞬間 —— Solmyr 順手又在勾上打了一個(gè)點(diǎn)。
“為什么打個(gè)點(diǎn)?”,zero 不滿的問。
“因?yàn)槟愕挠?jì)數(shù)器設(shè)計(jì)不佳,想象一下 Carrier ,它占 8 個(gè)人口,你是不是要在 Carrier 類中寫 8 個(gè) Counter 成員?或者聲明一個(gè) Counter 的數(shù)組?這樣的聲明清晰嗎?易讀嗎?”
“呃 ……”
“而且這樣使用 Counter 成員變量,需要計(jì)數(shù)的對象在空間上會付出更大的代價(jià),對于小對象,大小甚至可能翻一倍。”
“嗯 ……”
“更進(jìn)一步的說,計(jì)數(shù)值為 n 的對象,需要構(gòu)造 n 個(gè) Counter 對象,運(yùn)行性能也要受影響。”
“啊 …… ”
“現(xiàn)在你說說看,怎么改進(jìn)你的計(jì)數(shù)器,同時(shí)不用改動原來的客戶代碼?”
“哦 …… ”
zero 陷入了沉思:改進(jìn)后的計(jì)數(shù)器應(yīng)該有指定計(jì)數(shù)值的能力,這個(gè)能力應(yīng)該是 …… 應(yīng)該是對應(yīng)于一個(gè)計(jì)數(shù)器對象而非整個(gè)計(jì)數(shù)器類的,因?yàn)楣蚕硗粋€(gè)計(jì)數(shù)器的類可能計(jì)數(shù)值不同,也就是說這里需要為計(jì)數(shù)器類的對象指定一個(gè)參數(shù) …… 啊!原來這么簡單!
“我知道了!答案就是構(gòu)造函數(shù)!”,zero 飛快的把計(jì)數(shù)器類的定義改為(原來定義請參見上一期):
template <class T>
class Counter
{
public:
Counter(int step) // 改動部分
{
m_step = step;
m_count += m_step;
};
~Counter(){ m_count -= m_step; }; // 改動部分
int GetCout(){ return m_count; };
private:
static int m_count;
int m_step; // 新加部分
};
“嗯,不錯(cuò),不過還有問題。”,Solmyr 一邊點(diǎn)頭一邊說,“這樣一來,以前編寫的使用 Counter 類的客戶代碼就不能編譯了 —— 它們會報(bào)告說構(gòu)造的時(shí)候少了一個(gè)參數(shù)。”
“這好辦。”,zero 很快發(fā)現(xiàn)了自己漏掉了什么。他把構(gòu)造函數(shù)的定義改為:
Counter(int step = 1)
“這樣一來,以前的客戶代碼會缺省的得到計(jì)數(shù)值 1 ,就像以前一樣。”
“嗯,表現(xiàn)不錯(cuò),不過 ……”
zero 心中一緊。
“算了,今天就這樣吧。”
“Yeah!”
“把今天這些討論整理成詳細(xì)文檔,下班以前交給我”
“啊!~~~~”
………………
就這樣,再一次的,故事在 zero 的慘叫聲中結(jié)束了。