C++類成員屬性的一種簡(jiǎn)潔實(shí)現(xiàn)
? ? 一般來(lái)說(shuō)對(duì)于標(biāo)準(zhǔn)C++而言是不存在成員屬性這個(gè)概念的,以前大家都是用GetXXX/SetXXX來(lái)訪問(wèn)或取得數(shù)據(jù),好象也沒(méi)有感覺(jué)到任何不便。但是當(dāng)我們用過(guò)C#之類的語(yǔ)言之后,我們總覺(jué)得C++這個(gè)方式太老土了。于是我們想去實(shí)現(xiàn)“屬性”這個(gè)C++語(yǔ)言缺乏的要素。事實(shí)上網(wǎng)絡(luò)上有很多人已經(jīng)做了這部分工作,實(shí)現(xiàn)的方法有很多種,一種是用模板,一種是根據(jù)特定語(yǔ)言來(lái)寫(xiě)的,如VC(指的是Microsoft實(shí)現(xiàn)的C++)。但是它們要么很復(fù)雜,要么很難記住它的準(zhǔn)確用法,嗯我總是喜歡簡(jiǎn)單的東西,因?yàn)樘珡?fù)雜的東東會(huì)讓我的頭腦當(dāng)機(jī)。廢話少說(shuō),來(lái)看看如何實(shí)現(xiàn)。
在實(shí)現(xiàn)之前,我必需先探討一下為什么需要“屬性”這個(gè)東東。比如說(shuō)下面雇員這個(gè)類:
class?CEmployee{public:? int?Old;?//年齡};CEmployee?employee;employee.Old=22;int?old?=employee.Old;
? ? 它有一個(gè)成員變量,我們可以直接對(duì)它們進(jìn)行賦值或者讀取,但是往往會(huì)缺少一個(gè)很重要的東東,就是不能對(duì)所賦值進(jìn)行校驗(yàn),這可是個(gè)大問(wèn)題,比如我們給Old一個(gè)負(fù)值,比如-50,程序運(yùn)行時(shí)不會(huì)有任何錯(cuò)誤,但是的確這個(gè)成員變量的值在邏輯上是不正確的。于是我們會(huì)寫(xiě)上GetOld、SetOld?,F(xiàn)在OK了,這個(gè)小問(wèn)題解決了,但新問(wèn)題來(lái)了。我們的類使用者,他們需要重新把他們的代碼成寫(xiě)如下的樣子,而不是上面的那樣。
??CEmployee?employee;??employee.SetOld(22);??int?old?=employee.GetOld();
? ? 你的伙伴一定會(huì)在寫(xiě)代碼時(shí)詛咒你寫(xiě)了一個(gè)垃圾的類。所以你決定要改變這個(gè)現(xiàn)狀。很幸運(yùn),你是MS的忠實(shí)用戶,而且你對(duì)于MSDN看很仔細(xì),所以你知道可以這樣來(lái)寫(xiě)
class?CEmployee{private:??int?m_old;public:?_declspec(property(get=?GetOld,put=SetOld))int?Old;??int?GetOld(void)?{??return?m_old;?}?void?SetOld(int?value)?{??if(?(value?>0)?&&?(value?<60))??{??m_old?=?value;??}??else??{??m_old?=20;??}?}};
? ? Very Good,上面的類完美地完成一個(gè)屬性所要做的目標(biāo),不過(guò)還有一點(diǎn)小問(wèn)題,象我這樣比較笨的經(jīng)常需要查找MSDN才會(huì)知道?
?
_declspec(property(get=?GetOld,put=SetOld))int?Old;
? ? 這句話的含義,而且我也經(jīng)常忘記它的具體寫(xiě)法,比如put我常把它寫(xiě)成了set,這總是讓我想起了使用C#的美好時(shí)光,它是可以寫(xiě)成這個(gè)樣子的
public?class?CEmployee?{??private?int?m_old;??public?int?Old??{??get??{?return?m_old;?}??set??{?? ?if(value?>0?&&?value?<60)? ? {? ? ? ? m_old = value;? ? ?}?? ?else? ? {? ? ? ? m_old =20;? ? ?}??}??}?}
所以我想到可以利用C/C++中強(qiáng)大的武器宏,我們來(lái)定義幾個(gè)宏
#define?PROP(T,X)?__declspec(property(get=?__get##X,put=?__put##X))T?X;#define?GETPROP(T,X)?__declspec(property(get=?__get##X))T?X;?//只讀屬性#define?SETPROP(T,X)?__declspec(property(put=?__put##X))T?X;?//只寫(xiě)屬必#define?GET(T,X)?T?__get##X(void)?#define?SET(T,X)?void?__put##X(T?value)
? ? 說(shuō)明一下:T 代表屬性的類型如int,double,CString,而X代表屬性名稱。如果你需要一個(gè)只讀屬性可以使用GETPROP,只寫(xiě)屬性則可以使用SETPORP,然后對(duì)應(yīng)使用一個(gè)GET或SET,當(dāng)然如果你用PROP,而只用了一個(gè)GET或SET,也沒(méi)有錯(cuò),只是在編譯時(shí)會(huì)告訴你沒(méi)有一個(gè)__getXXX或__putXXX的方法。然后我們就可以這樣來(lái)寫(xiě)我們的類。
class?CEmployee{private:??int?m_old;public:?PROP(int?,Old)?GET(int,Old)?{??return?m_old;?}?SET(int,Old)?{??if(?(value?>0)?&&?(value?<60))?//這里的value你可把它和C#一樣當(dāng)做關(guān)鍵字??{??m_old?=?value;??}??else??{??m_old?=20;??}?}};
? ? 好了,我們要做的工作已經(jīng)做完了。當(dāng)然這種方法還是有很多問(wèn)題,比如不能使用C#中常用的索引屬性,靜態(tài)屬性等等。但是畢竟我們是C++程序員么,呵呵!最后,這種方法只是在VC下有用。