[導(dǎo)讀]命名空間、靜態(tài)成員、友元函數(shù)及運算符重載
前言
在上一則教程中,著重地闡述了構(gòu)造函數(shù)以及析構(gòu)函數(shù)的相關(guān)概念,這也是C 中非常重要的兩個概念之一。在今天的教程中,筆者將繼續(xù)敘述?C 相對于?C語言來說不同的點,將詳細(xì)敘述命名空間,靜態(tài)成員,友元函數(shù)以及運算符重載這幾個知識點。C 命名空間
命名空間的存在是為了區(qū)分不同庫的相同的函數(shù)名,用一個簡單的例子來說明這個問題就是在?windows的文件系統(tǒng)中,不同文件夾下可以有相同名字的文件,相同文件夾下因為這相同文件處在不同的范圍內(nèi),用 C 說白了也就是處在不同的命名空間中。文件系統(tǒng)的一個結(jié)構(gòu)圖:定義命名空間
命名空間的定義使用的是關(guān)鍵字 namespace,后跟命名空間的名稱,如下所示:namespace?namespace_name{
????//?代碼聲明
}
為了調(diào)用帶有命名空間的函數(shù)或者變量,需要在前面加上命名空間的名稱,如下所示:name::code???//?code?可以是變量或者是函數(shù)
例子
下面通過一個例子來說明命名空間的概念,首先,我們具有兩個類,一個是 Dog ,一個是 Person,而這個時候,有兩個函數(shù)具有相同的名字,都要輸出不同的信息,這個時候,就有必要使用到命名空間的概念。首先,我們在 dog.h 里面定義一個 dog 類,代碼如下所示:#ifndef?__DOG_H__
#define?__DOG_H__
namespace?C{
class?Dog{
private:
????char?*name;
????int?age;
public:
????void?setName(char?*name);
????int?setAge(int?age);
????void?printInfo(void);
};
void?printVersion(void);
}
#endif
然后,緊接著來看 dog.cpp 里面的內(nèi)容。代碼如下所示:#include?"dog.h"
namespace?C{
????void?Dog::setName(char?*name)
????{
????????this->name?=?name;
????}
????int?Dog::setAge(int?age)
????{
????????if?(age?0?||?age?>?20)
????????{
????????????this->age?=?0;
????????????return?-1;
????????}
????????this->age?=?age;
????????return?0;
????}
????void?Dog::printInfo(void)
????{
????????printf("name?=?%s,?age?=?%d\n",name,age);
????}
????void?printersion(void)
????{
????????printf("Dog?v1");
????}
}
OK ,看完了 Dog 的代碼,我們緊接著來看 Person 的代碼,代碼如下所示:#ifndef?__PERSON_H__
#define?__PERSON_H__
namespace?A{
class?Person{
private:
????char?*name;
????int?age;
????char?*work;
public:
????void?setName(char?*name);
????int?setAge(int?age);
????void?printInfo(void);
????};
????void?printfVersion(void);
}
#endif
緊接著就是 Person.cpp 的代碼,具體的代碼如下所示:namespace?A?{
void?Person::setName(char?*name)
{
????this->name?=?name;
}
int?Person::setAge(int?age)
{
????if?(age?0?||?age?>?150)
????{
????????this->age?=?0;
????????return?-1;
????}
????this->age?=?age;
????return?0;
}
void?Person::printInfo(void)
{
????printf("name?=?%s,?age?=?%d,?work?=?%s\n",?name,?age,?work);?
}
void?printVersion(void)
{
????printf("Person?v1\n");
}
}
上述就是 所定義的兩個類,我們緊接著來看 main.cpp 的代碼:int?main(int?argc,?char?**argv)
{
????A::Person?per;
????per.setName("zhangsan");
????per.setAge(16);
????per.printInfo();
????C::Dog?dog;
????dog.setName("wangcai");
????dog.setAge(1);
????dog.printInfo();
????A::printVersion();
????C::printVersion();
????return?0
}
在最后的倒數(shù)第二行和倒數(shù)第三行,我們可以看到如果這個時候,沒有命名空間的存在,那么就完全不能夠分辨?printVersion這個函數(shù),加上了命名空間之后,就能夠分辨出來了。靜態(tài)成員
在上述代碼的基礎(chǔ)上,我們在主函數(shù)定義了如何幾個變量,代碼如下所示:#include?
int?main(int?argc,?char?**argv)
{
????Person?per1;
????Person?per2;
????Person?per3;
????Person?per4;
????Person?*per5?=?new?Person[10];
}
那我們要如何知道我們定義幾個 Person 對象呢,可以這樣去做,我們創(chuàng)建一個?cnt變量,然后在每個構(gòu)造函數(shù)執(zhí)行的過程中讓?cnt加一,代碼如下所示:#include?
#include?
#include?
class?Person
{
private:
????int?cnt;
????char?*name;
????int?age;
????char?*work;
public:
????Person()
????{
????????name?=?NULL;
????????work?=?NULL;
????????cnt ;
????}
????Person(char?*name)
????{
????????this->name?=?new?char[strlen(name)? ?1];
????????strcpy(this->name,?name);
????????this->work?=?NULL;
????????cnt ;
????}
????Person(char?*name,?int?age,?char?*work?=?"none")
????{
????????this->name?=?new?char[strlen(name)? ?1];
????????strcpy(this->name,?name);
????????this->work?=?new?char[strlen(work)? ?1];
????????strcpy(this->work,?work);
????????cnt ;
????}
????~Person()
????{
????????if?(this->name)
????????{
????????????cout?<"name?is:"?<endl;
????????????delete?this->name;
????????}
????????if?(this->work)
????????{?
????????????cout?<"work?is:"?<endl;
????????????delete?this->work;
????????}
????}
};
但是如果這么寫的話存在一個問題,就是我們想要實現(xiàn)的功能是看有幾個實例化?Person?對象,那么這個計數(shù)量cnt應(yīng)該是屬于?Person類的,具體的關(guān)系如下圖所示:cnt是屬于?Person的實例化對象的,那要如何做才能使得?cnt屬于?Person類的實例化對象呢,這個時候,我們需要將?cnt定義為?static類的,這樣子,cnt就是屬于?Person類的了,定義的代碼如下所示:class?Person
{
private:
????char?*name;
????int?age;
????char?*work;
????static?int?cnt;
};
那么我們要如何得到 cnt 的值呢,可以編寫一個函數(shù),但是同樣的,我們編寫的函數(shù)要是屬于整個?Person類的,那應(yīng)該如何去做呢,同樣的辦法,我們在前面加上?static,代碼如下所示:#include?
#include?
class?Person
{
private:
????char?*name;
????int?age;
????char?*work;
????static?int?cnt;
public:
????static?int?getcount(void)
????{
????????return?cnt;
????}
};
有了?getcount函數(shù),我們就可以調(diào)用它,然后將其打印出來,方法如下所示:#include?
int?main(int?argc,?char?*argv)
{
????Person?per1;
????Person?per2;
????Person?*per5?=?new?Person[10];
????count?<"person?number?=?"?<endl;
}
最后,還存在一個問題,因為我們在?cnt上加了?static,那么當(dāng)前的?cnt就是屬于?Person類的,這樣一來,那么就是說?cnt的值還沒有分配空間,那么要如何分配空間呢,我們需要在主函數(shù)開始之前對?cnt進(jìn)行定義和初始化,代碼如下所示:int?Person::cnt?=?0;?????/*?定義*/
這樣的話,就可以知道?cnt的值了,下面是運行的結(jié)果:Person?類的實例化次數(shù)。那為什么要把?int Person::cnt = 0放在?main函數(shù)的最開始呢,這是因為要在?main所有實例化對象定義之前就要將其初始化完成。友元函數(shù)
首先,我們有這樣一個需求,需要實現(xiàn)兩個類的相加,下面是寫出來的代碼:#include?
#include?
#include?
using?namespace?std;
class?Point
{
private:
????int?x;
????int?y;
public:
????Point(){}
????Point(int?x,?int?y)?:?x(x),?y(y)?{}
????void?setX(int?x)
????{
????????this->x?=?x;
????}
????void?setY(int?y)
????{
????????this->y?=?y;
????}
????int?getX(void)
????{
????????return?x;
????}
????int?getY(void)
????{
????????return?y;
????}
};
Point?add(Point 




