欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++特殊類設(shè)計(jì)概念與示例講解

 更新時(shí)間:2023年04月22日 11:27:51   作者:命由己造~  
本文介紹C++中三種特殊類設(shè)計(jì)模式:單例模式、工廠模式和代理模式。通過詳細(xì)講解每種設(shè)計(jì)模式的實(shí)現(xiàn)原理和應(yīng)用場景,幫助讀者理解和掌握這些常用的面向?qū)ο笤O(shè)計(jì)模式,并提供示例代碼和技巧,便于實(shí)際應(yīng)用

一、設(shè)計(jì)模式概念

設(shè)計(jì)模式是一套被反復(fù)使用、多數(shù)人知曉的、經(jīng)過分類的、代碼設(shè)計(jì)經(jīng)驗(yàn)的總結(jié)。

使用設(shè)計(jì)模式的目的:為了代碼可重用性、讓代碼更容易被他人理解、保證代碼可靠性。

根本原因是為了代碼復(fù)用,增加可維護(hù)性。

設(shè)計(jì)模式的例子:迭代器模式

二、設(shè)計(jì)一個(gè)不能被拷貝的類

拷貝一共就只有兩個(gè)場景,一個(gè)是拷貝構(gòu)造,一個(gè)是賦值運(yùn)算符重載。所以我們想要設(shè)計(jì)出一個(gè)不能被拷貝的類只需要讓外部無法調(diào)用這兩個(gè)函數(shù)即可。

在C++98中,我們的方法是將拷貝構(gòu)造和賦值運(yùn)算符重載只聲明不定義并且將權(quán)限設(shè)置為私有。

class anti_copy
{
public:
	anti_copy()
	{}
private:
	anti_copy(const anti_copy& ac);
	anti_copy& operator=(const anti_copy& ac);
};

設(shè)計(jì)原因:

1?? 私有:如果聲明成共有,那么就可以在類外面實(shí)現(xiàn)定義。

2?? 只聲明不定義:因?yàn)槿绻宦暶骶幾g器會(huì)默認(rèn)生成這兩個(gè)的默認(rèn)成員函數(shù)。而不定義是因?yàn)樵摵瘮?shù)不會(huì)被調(diào)用,就不用寫了,這樣編譯的時(shí)候就會(huì)出現(xiàn)鏈接錯(cuò)誤。

而在C++11中引入了關(guān)鍵字——delete。

如果在默認(rèn)成員函數(shù)后跟上=delete,表示讓編譯器刪除掉該默認(rèn)成員函數(shù)。即使權(quán)限是共有也無法調(diào)用已刪除的函數(shù)。

class anti_copy
{
public:
	anti_copy()
	{}
	anti_copy(const anti_copy& ac) = delete;
	anti_copy& operator=(const anti_copy& ac) = delete;
private:
};

三、設(shè)計(jì)一個(gè)只能在堆上創(chuàng)建對(duì)象的類

3.1 私有構(gòu)造

首先要把構(gòu)造函數(shù)給私有,不然這個(gè)類就可以在任意位置被創(chuàng)建。而構(gòu)造函數(shù)被私有了以后我們怎么創(chuàng)建對(duì)象呢?

我們可以在定義一個(gè)成員函數(shù),讓這個(gè)函數(shù)在堆上申請空間,但我們知道必須現(xiàn)有對(duì)象才能調(diào)用成員函數(shù)。所以我們就把這個(gè)函數(shù)設(shè)置成靜態(tài)成員函數(shù)。

class OnlyHeap
{
public:
	static OnlyHeap* GetObj()
	{
		return new OnlyHeap;
	}
private:
	OnlyHeap()
	{}
};

但是這樣也不完全對(duì),如果我們這么寫:

class OnlyHeap
{
public:
	static OnlyHeap* GetObj()
	{
		return new OnlyHeap;
	}
private:
	OnlyHeap()
	{}
};
int main()
{
	OnlyHeap* hp1 = OnlyHeap::GetObj();
	OnlyHeap hp2(*hp1);
	return 0;
}

這里的hp2就是棧上的對(duì)象。所以我們也要把拷貝構(gòu)造給封住。

class OnlyHeap
{
public:
	static OnlyHeap* GetObj()
	{
		return new OnlyHeap;
	}
	OnlyHeap(const OnlyHeap& hp) = delete;
private:
	OnlyHeap()
	{}
};

3.2 私有析構(gòu)

class OnlyHeap
{
public:
	OnlyHeap()
	{}
	OnlyHeap(const OnlyHeap& hp) = delete;
private:
	~OnlyHeap()
	{}
};
int main()
{
	OnlyHeap hp1;// error
	OnlyHeap* hp2 = new OnlyHeap;
	return 0;
}

這里的hp1就不能創(chuàng)建成功,因?yàn)閷?duì)象銷毀的時(shí)候會(huì)調(diào)用析構(gòu)函數(shù),但是這里的析構(gòu)是私有的,所以該對(duì)象無法調(diào)用。

但是我們要銷毀hp2該怎么辦呢?

我們可以定義一個(gè)成員函數(shù)顯示調(diào)用析構(gòu)函數(shù)。

class OnlyHeap
{
public:
	OnlyHeap()
	{}
	OnlyHeap(const OnlyHeap& hp) = delete;
	void Destroy()
	{
		this->~OnlyHeap();
	}
private:
	~OnlyHeap()
	{}
};
int main()
{
	OnlyHeap* hp2 = new OnlyHeap;
	hp2->Destroy();
	return 0;
}

四、設(shè)計(jì)一個(gè)只能在棧上創(chuàng)建對(duì)象的類

為了不讓這個(gè)類隨便定義出對(duì)象,首先要把構(gòu)造函數(shù)私有。然后跟上面只能在堆上創(chuàng)建對(duì)象的方法相似,定義出一個(gè)靜態(tài)成員函數(shù)返回棧上創(chuàng)建的對(duì)象。

class StackOnly
{
public:
	static StackOnly GetObj()
	{
		return StackOnly();
	}
private:
	StackOnly()
	{}
};
int main()
{
	StackOnly hp = StackOnly::GetObj();
	return 0;
}

但是這里有一個(gè)問題,無法防止創(chuàng)建靜態(tài)對(duì)象:

static StackOnly hp2 = StackOnly::GetObj();

五、設(shè)計(jì)不能被繼承的類

在C++98,為了不讓子類繼承,我們可以把構(gòu)造函數(shù)私有化,因?yàn)樽宇愋枰日{(diào)用父類的構(gòu)造函數(shù)初始化父類的那一部分成員。

class NoInherit
{
public:
private:
	NoInherit()
	{}
};

而在C++11中引入的新的關(guān)鍵字final,被final關(guān)鍵字修飾的類不能被繼承。

class NoInherit final
{
public:
private:
};

六、單例模式

一個(gè)類只能創(chuàng)建一個(gè)對(duì)象,即單例模式,該模式可以保證系統(tǒng)中該類只有一個(gè)實(shí)例,并提供一個(gè)訪問它的全局訪問點(diǎn),該實(shí)例被所有程序模塊共享。

單例模式的特點(diǎn)就是全局只有一個(gè)唯一對(duì)象。

6.1 餓漢模式

怎么能做到全局只是用一個(gè)對(duì)象呢,比方說我們現(xiàn)在想要實(shí)現(xiàn)一個(gè)英漢字典,首先我們要把構(gòu)造函數(shù)私有,不然無法阻止創(chuàng)建對(duì)象。然后我們可以在類里面定義一個(gè)自己類型的靜態(tài)成員變量,作用域是全局的。因?yàn)閷?duì)比定義在外邊的靜態(tài)成員變量,內(nèi)部的可以調(diào)用構(gòu)造函數(shù)。

這里要注意把拷貝也要封住。

class Singleton
{
public:
	static Singleton& GetObj()
	{
		return _s;
	}
	void insert(const std::string& s1, const std::string& s2)
	{
		_dict[s1] = s2;
	}
	void Print()
	{
		for (auto& e : _dict)
		{
			cout << e.first << "->" << e.second << endl;
		}
	}
	// 防拷貝
	Singleton(const Singleton&) = delete;
	Singleton& operator=(const Singleton&) = delete;
private:
	Singleton()
	{}
	std::map<std::string, std::string> _dict;
private:
	static Singleton _s;// 聲明
};
Singleton Singleton::_s;// 定義
int main()
{
	Singleton::GetObj().insert("corn", "玉米");
	Singleton& dic1 = Singleton::GetObj();
	dic1.insert("apple", "蘋果");
	dic1.insert("banana", "香蕉");
	Singleton& dic2 = Singleton::GetObj();
	dic2.insert("pear", "梨");
	dic2.Print();
	return 0;
}

餓漢模式有什么特點(diǎn)呢?

它會(huì)在一開始(main之前)就創(chuàng)建對(duì)象。

餓漢模式有什么缺點(diǎn)呢?

1?? 如果單例對(duì)象構(gòu)造十分耗時(shí)或者占用很多資源,比如加載插件啊, 初始化網(wǎng)絡(luò)連接啊,讀取文件啊等等,而有可能該對(duì)象程序運(yùn)行時(shí)不會(huì)用到,那么也要在程序一開始就進(jìn)行初始化,就會(huì)導(dǎo)致程序啟動(dòng)時(shí)非常的緩慢。

2?? 多個(gè)單例類之間如果有依賴關(guān)系餓漢模式就無法控制,比方說要求A類初始化時(shí)必須調(diào)用B,但是餓漢無法控制先后順序。

所以針對(duì)這些問題,就有了懶漢模式。

6.2 懶漢模式

第一次使用實(shí)例對(duì)象時(shí),創(chuàng)建對(duì)象(用的時(shí)候創(chuàng)建)。進(jìn)程啟動(dòng)無負(fù)載。多個(gè)單例實(shí)例啟動(dòng)順序自由控制。

我們可以直接對(duì)上面餓漢模式的代碼進(jìn)行修改,把靜態(tài)成員變量變成指針。然后把獲取的函數(shù)改變一下:

static Singleton& GetObj()
	{
		// 第一次調(diào)用才會(huì)創(chuàng)建對(duì)象
		if (_s == nullptr)
		{
			_s = new Singleton;
		}
		return *_s;
	}

整體代碼:

class Singleton
{
public:
	static Singleton& GetObj()
	{
		// 第一次調(diào)用才會(huì)創(chuàng)建對(duì)象
		if (_s == nullptr)
		{
			_s = new Singleton;
		}
		return *_s;
	}
	void insert(const std::string& s1, const std::string& s2)
	{
		_dict[s1] = s2;
	}
	void Print()
	{
		for (auto& e : _dict)
		{
			cout << e.first << "->" << e.second << endl;
		}
	}
	// 防拷貝
	Singleton(const Singleton&) = delete;
	Singleton& operator=(const Singleton&) = delete;
private:
	Singleton()
	{}
	std::map<std::string, std::string> _dict;
private:
	static Singleton* _s;// 聲明
};
Singleton* Singleton::_s = nullptr;// 定義

6.2.1 線程安全問題

上面的代碼存在問題,當(dāng)多個(gè)線程同時(shí)調(diào)用GetObj(),就會(huì)創(chuàng)建多個(gè)對(duì)象。所以為了線程安全我們要加鎖。為了保證鎖自動(dòng)銷毀,我們可以自定義一個(gè)鎖。

template <class Lock>
class LockAuto
{
public:
	LockAuto(Lock& lk)
		: _lk(lk)
	{
		_lk.lock();
	}
	~LockAuto()
	{
		_lk.unlock();
	}
private:
	Lock& _lk;
};
class Singleton
{
public:
	static Singleton& GetObj()
	{
		// 第一次調(diào)用才會(huì)創(chuàng)建對(duì)象
		if (_s == nullptr)// 只有第一次才用加鎖
		{
			LockAuto<mutex> lock(_mutex);
			if (_s == nullptr)
			{
				_s = new Singleton;
			}
		}
		return *_s;
	}
	void insert(const std::string& s1, const std::string& s2)
	{
		_dict[s1] = s2;
	}
	void Print()
	{
		for (auto& e : _dict)
		{
			cout << e.first << "->" << e.second << endl;
		}
	}
	// 防拷貝
	Singleton(const Singleton&) = delete;
	Singleton& operator=(const Singleton&) = delete;
private:
	Singleton()
	{}
	std::map<std::string, std::string> _dict;
private:
	static Singleton* _s;// 聲明
	static mutex _mutex;// 鎖
};
Singleton* Singleton::_s = nullptr;// 定義
mutex Singleton::_mutex;// 定義

6.2.2 新寫法

class Singleton
{
public:
	static Singleton& GetObj()
	{
		static Singleton dic;
		return dic;
	}
	void insert(const std::string& s1, const std::string& s2)
	{
		_dict[s1] = s2;
	}
	void Print()
	{
		for (auto& e : _dict)
		{
			cout << e.first << "->" << e.second << endl;
		}
	}
	// 防拷貝
	Singleton(const Singleton&) = delete;
	Singleton& operator=(const Singleton&) = delete;
private:
	Singleton()
	{}
	std::map<std::string, std::string> _dict;
};

這里就用了靜態(tài)局部變量只會(huì)在第一次定義的時(shí)候初始化。在C++11之前是不能保證線程安全的,但是C++11之后就可以了。

到此這篇關(guān)于C++特殊類設(shè)計(jì)概念與示例講解的文章就介紹到這了,更多相關(guān)C++特殊類設(shè)計(jì)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++的cout.tellp()和cout.seekp()語法介紹

    C++的cout.tellp()和cout.seekp()語法介紹

    無論是使用 cout 輸出普通數(shù)據(jù),用 cout.put() 輸出指定字符,還是用 cout.write() 輸出指定字符串,數(shù)據(jù)都會(huì)先放到輸出流緩沖區(qū),待緩沖區(qū)刷新,數(shù)據(jù)才會(huì)輸出到指定位置,本文給大家介紹一下C++的cout.tellp()和cout.seekp()語法,需要的朋友可以參考下
    2023-09-09
  • OpenCV利用霍夫變換實(shí)現(xiàn)交通車道線檢測

    OpenCV利用霍夫變換實(shí)現(xiàn)交通車道線檢測

    經(jīng)典霍夫變換用來檢測圖像中的直線,后來霍夫變換經(jīng)過擴(kuò)展可以進(jìn)行任意形狀物體的識(shí)別,例如圓和橢圓。本文就來利用霍夫變換實(shí)現(xiàn)交通車道線檢測,需要的可以參考一下
    2022-09-09
  • 使用C++模擬實(shí)現(xiàn)2024春晚劉謙魔術(shù)

    使用C++模擬實(shí)現(xiàn)2024春晚劉謙魔術(shù)

    劉謙在2024年春晚上的撕牌魔術(shù)的數(shù)學(xué)原理非常簡單,所以這篇文章主要為大家詳細(xì)介紹了如何使用C++模擬實(shí)現(xiàn)這一魔術(shù)效果,感興趣的可以了解下
    2024-02-02
  • 一文帶你學(xué)會(huì)C語言中的qsort函數(shù)

    一文帶你學(xué)會(huì)C語言中的qsort函數(shù)

    qsort函數(shù)是C語言的庫函數(shù),能實(shí)現(xiàn)對(duì)各種元素類型的比較,使用的基本思想是快速排序法,頭文件是<stdlib.h>,本文不講解具體實(shí)現(xiàn)原理,只對(duì)使用方法進(jìn)行說明,希望對(duì)大家有所幫助
    2022-12-12
  • C++抽象數(shù)據(jù)類型介紹

    C++抽象數(shù)據(jù)類型介紹

    這篇文章主要介紹了C++抽象數(shù)據(jù)類型,我們在學(xué)數(shù)據(jù)結(jié)構(gòu)的時(shí)候,經(jīng)常遇到的一個(gè)概念就是抽象數(shù)據(jù)類型(Abstract Data Type),簡稱ADT。下面我們就對(duì)ADT作更多介紹,需要的朋友可以參考一下
    2022-01-01
  • C/C++編程判斷String字符串是否包含某個(gè)字符串實(shí)現(xiàn)示例

    C/C++編程判斷String字符串是否包含某個(gè)字符串實(shí)現(xiàn)示例

    這篇文章主要為大家介紹了C++編程中判斷String字符串是否包含某個(gè)字符串的實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2021-11-11
  • C語言輪轉(zhuǎn)數(shù)組的三種實(shí)現(xiàn)

    C語言輪轉(zhuǎn)數(shù)組的三種實(shí)現(xiàn)

    輪轉(zhuǎn)數(shù)組是一種將數(shù)組元素循環(huán)移動(dòng)的處理方式,它通常用于解決一些需要對(duì)固定長度的數(shù)組進(jìn)行循環(huán)滾動(dòng)處理的問題,本文就介紹了C語言輪轉(zhuǎn)數(shù)組的三種實(shí)現(xiàn),感興趣的可以了解一下
    2023-08-08
  • C++ 漢諾塔問題知識(shí)點(diǎn)總結(jié)

    C++ 漢諾塔問題知識(shí)點(diǎn)總結(jié)

    在本篇文章里小編給大家整理的是關(guān)于C++ 漢諾塔問題知識(shí)點(diǎn)內(nèi)容,有需要的朋友們可以參考下。
    2020-02-02
  • 基于C語言實(shí)現(xiàn)簡單的12306火車售票系統(tǒng)

    基于C語言實(shí)現(xiàn)簡單的12306火車售票系統(tǒng)

    火車售票系統(tǒng)給我們的出行帶來了極大的方面,那么他基于編程是如何實(shí)現(xiàn)的呢?今天小編抽時(shí)間給大家分享一個(gè)使用C語言寫的一個(gè)簡單的火車票系統(tǒng),感興趣的朋友參考下
    2016-09-09
  • 關(guān)于C語言和命令行之間的交互問題

    關(guān)于C語言和命令行之間的交互問題

    這篇文章主要介紹了C語言和命令行之間的交互,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-07-07

最新評(píng)論