芥子須彌----封裝(轉)
發(fā)表時間:2023-07-18 來源:明輝站整理相關軟件相關文章人氣:
[摘要]芥子須彌----封裝作者:HolyFire說起面向對象OOP首先就有人大喊著封裝究竟封裝是什么呢,為什么要封裝?是的,本來并沒有封裝,封裝是由于人們的需要才產生的,就如同計算機來到人間,編程語言進入...
芥子須彌
----封裝
作者:HolyFire
說起面向對象OOP首先就有人大喊著封裝
究竟封裝是什么呢,為什么要封裝?
是的,本來并沒有封裝,封裝是由于人們的需要才產生的,就如同計算機來到人間,編程語言進入你的大腦,自然而然。
在沒有封裝的時候,人們一樣在編程,并沒有因為封裝的出現(xiàn)使得猿猴變成了人,所以封裝沒有那么神秘,可怕。
想象一下,我們生活里也有很多封裝:食品被放在袋子里,用來防止混入灰塵和昆蟲侵蝕;戴上太陽鏡,防止紫外線傷害眼睛;錢和貴重物品放在保險柜里,防止心術不正的人。
好了,我們看出一點,封裝的一個作用是保護我們的東西。
在生活中,有很多慣例,這里要提到一點,那就永真式,這是一個表示在無論什么情況下都為真的式子,當然這是一種理想的情況,但人們往往喜歡創(chuàng)造這樣的東西,就象“水往低處流” 。牛頓說“由于地球引力,所以蘋果是往地上掉的”幸虧有這樣的道理,我們才有水喝,有蘋果吃。當然了在地球上很難找出反例,所以我們無需考慮蘋果飛上天的可能性。又如人的聽覺范圍是2kHZ~~20kHZ,如果有人說地球上會有一個人的聽覺超出這個范圍,我一點異議都沒有,但是我還是把人的聽覺作為2kHZ~~20kHZ來處理,應為概率論上有一句,小概率的事是不會發(fā)生的,我不打算證實他,我只是想利用他,我只處理2kHZ~~20kHZ的聽覺范圍,在絕大多數(shù)領域里,這樣做足夠了,讓在這方面有特別要求的人掙扎去吧,少處理這些個別的人會帶來更多的好處。
我們又能發(fā)覺到,封裝的好處能降低復雜度。
我想沒有人會愚蠢到將一只羊和一只牛相加,得到兩只羊或兩只牛這樣的結果,但是兩只動物這樣的結果卻是可以接受的。
這樣看來,封裝他能夠降低出錯的可能性。
封裝所做的遠遠不是這些。
等等,各位心中是不是充滿了疑惑。
講了這么久,為什么不說明封裝到底是什么呢,文中并沒有提到啊。
聰明的朋友,封裝并不是什么東西,也不是什么方法,他是人們考慮和解決問題的思路,他就是你的思想,就是你的靈感一閃,就是你的錦囊妙計。封裝就是人們?yōu)榱烁玫墓芾砗褪褂檬挛锏姆桨,它可以讓你更方便,更安全的做你想做的事,長久以來,充滿智慧的杰出者們摸索出一些使用封裝來解決問題的辦法。
在編程的應用上,我們看看封裝給我們帶來的好處吧。
由于計算機里,一些都數(shù)字化了,所以信息都是存放在很多很多的存儲單元里的,由于硬件的限制,這些存儲單元都有確實的大小。
首先我們來談談保護我們的數(shù)據(jù)不被隨便訪問,這正是我提到的。
比如說一個員工可以看到自己的工資,但如果他能夠修改的話,只怕人人都想去這個公司工作了。修改工資的數(shù)據(jù),萬萬不行,萬萬不行,但是如果人人都不能修改工資這一數(shù)據(jù),那也不行啊,財會需要修改關于工資的數(shù)據(jù)。
這里我使用面向對象開發(fā)而設計的C++語言來實現(xiàn)他,因為他是為面向對象設計的,所以實現(xiàn)起來比較方便。沒學過C++的人先要看一看基本概念,至少要理解private,public,protected,friend等關鍵字的概念。
我們來理解一下我們要做的事,公司里有很多人,人人都有工資,一些是員工,員工用工號來區(qū)別,一些是財會,而財會也是員工,員工能察看工資,而財會還能修改工資。
我們有三個角色,人,員工,財會,要做兩件事,員工察看工資,財會修改工資,然后理清他們的關系
人有姓名,年齡,性別 這三樣是我們需要處理的,我們要做的是處理工資,一些無關的信息就可以省略,習慣上每個部門都需要人的這三個信息
人 = { 姓名 , 年齡 , 性別 }
員工是人,財會是人,財會是員工,員工有工資,為了區(qū)別員工每個員工有工號
員工察看工資,財會修改工資
員工 ={ 人 , 工資 ,工號 , 察看工資 }
財會 ={ 員工 , 修改工資 }
//下面三句代碼是使用標準函數(shù)庫,可以節(jié)省我們很多功夫,方便我們理解我們要理解的,略過繁枝小節(jié)
#include <iostream>
#include <string>
using namespace std;
class Man{
public:
enum SexType { Mele , Female };//枚舉類型,性別只有男女兩種,這里不考慮人妖,如果輸入不是這樣種類型,編譯器會報警。類型轉換中如果不是者兩個值,也會報警,這樣我們減少了誤輸入引起的錯誤。
private: //變量不能亂操作,設定為私有,只有類的成員函數(shù)才能操作,起到保護作用
string Name;//姓名,string是標準函數(shù)庫里提供的類,可以方便的處理字符串
unsigned Age; //年齡,由于年齡不可能是負數(shù),所以用unsigned表示,這樣可以防止邏輯錯誤
SexType Sex;//性別
public: //對外的接口,當然要開放了
string GetName( void ){ return Name; }//得到人的名字
void SetName( string const& name )//改變人的名字,這里預先檢查了名字是否合法
{
if( str == "" )
return;
Name = name;
}
unsigned GetAge( void ){ return Age; }//得到人的年齡
void SetAge( unsigned age )//改變人的年齡,由于員工50歲就退休了,所以50以下的才合法
{
if( age >= 50 )
return;
Age = age;
}
SexType GetSex( void ){ return Sex; } //得到人的性別
Void SetSex( SexType sex ){ Sex = sex }//改變人的性別
};
class Accountant;
class Employee : public Man{
private:
unsigned ID;//工號
protected://雖然要加以保護,但是他的后繼類財會要操作的
unsigned Pay; //工資
public:
unsigned GetID( void );//取得工號
void SetID( unsigned );//改變工號
unsigned GetPay( void ){ return Pay; }//察看工資
friend class Accountant;//由于財會能夠修改所有員工的工資,所以要將訪問權信托給財會
};
class Accountant : public Employee{
public:
void SetPay( unsigned pay ){ Pay = pay; } //改變自己的工資
void SetPay( Employee * man , unsigned pay ){ man->Pay = pay; }; //改變別人的工資
};
當然這是經(jīng)過精心設計后的封裝,簡化了結構,正因為一開始細心的分析,才使得設計可以輕松自如,歸根結底是由于思想正確,好了,封裝是一種思想,我們現(xiàn)在將他體現(xiàn)了出來。
再看這個例子里,工資的類型是unsigned,非負整數(shù),呵呵,大家都不愿意到工資為負的公司工作吧,這里簡化問題是從人們的慣例的角度出發(fā)的,如果你的老板考慮工資為負的情況,那么…^_^
現(xiàn)在一個粗心的財會不小心改錯了,他多敲了一個0,哇歐,請客請客,但這個財會可就慘了,這樣的好事不會發(fā)生,不準發(fā)生,老板青著臉狂吼著。
設計不得不加上一個工資的上限,沒辦法啦,現(xiàn)在國家規(guī)定的嗎。
class Accountant : public Employee{
enum MaxPay{ MAXPAY = 8000 };
public:
void SetPay( unsigned pay )
{
if( pay > MAXPAY )
return;
Pay = pay;
}
void SetPay( Employee * man , unsigned pay )
{
if( pay > MAXPAY )
return;
man->Pay = pay;
}
};
可以看出封裝的作用就是減少出錯的可能,方便靈活的運用類型
在上面的例子里我們看到,類class是由一些變量和函數(shù)組成的,這些變量和函數(shù)是類的一部分,我們稱之為成員,變量就是成員變量,函數(shù)當然就叫成員函數(shù)了。為什么要這樣呢,我們考慮一下,事物是由物質和運動組成的,表現(xiàn)物質的一面我們通常描述他的一些屬性,即他擁有什么,表現(xiàn)一個運動我們通常使用一個過程,要將一個事物的信息描述清楚就需要這兩樣東西。在長期的實踐中程序員達成一個共識,將事物的特性(也就是它擁有的)稱之為屬性,他能夠產生的行為稱之為方法,數(shù)字化以后就是成員變量和成員函數(shù),他們的組成的整體就是類(類型),這個類型將作為一個單獨的節(jié)點考慮,就像例子中的Man,我們不會說這是一個姓名,年齡等等的組合體,而是將他作為一個類--class Man考慮,從而簡化了問題。細小的事物組合成大的事物,大的事物組合成更大的事物,這樣下去,再大難題也可以化作小模塊來處理,這正是封裝誘人的地方和他的使命。
需要補充的是,既然我們把數(shù)據(jù)保護起來,那么如何讓用戶訪問這些數(shù)據(jù)就是一個問題了,在上面的例子中看出public:申明的方法,用戶是可以使用的,而我們正是通過這些方法將數(shù)據(jù)的信息告訴使用者,這里我們將描述方法的部分就叫做接口(在C++里,就是類中成員函數(shù)的聲明,用戶一般只對public:部分的接口感興趣,所以有人建議將public:部分的內容寫在顯眼的地方,比如靠類的頂部),也有人說是界面,也就是類和外界溝通和交流使用的渠道,所以接口是很重要的,他直接關系到你的類使用的方面。
而使用者使用類的某個接口的時候就象是通知這個類型使用某個行為,就象是傳遞一個消息給他一樣,我們把使用接口稱之為傳遞消息,而類被調用方法稱之為接受消息。
現(xiàn)在我們可以出定義:封裝就是將事物的內容和行為都隱藏在實現(xiàn)里,用戶不需要知道其內部實現(xiàn),這是大量程序員反復勞動后得出的一致結論。這樣的好處就是使用方便,易于維護,任何一樣都可以使程序員為之心動。當然我們不能保證高效,但是不意味著使用封裝就沒有高效的可能,如果在封裝的基礎上保證高效的話,我實在找不出理由來拒絕他。