C++ 中的單例模式(普通,2B,文藝)
一、普通Singleton
#include<iostream>
using namespace std;
class Singleton
{
public:
static Singleton* getInstance();
private:
static Singleton* instance;
Singleton()
{
cout<<"constructor\n";
// do something
};
~Singleton()
{
cout<<"destructor\n";
//do something
}
};
Singleton* Singleton::instance = NULL;
Singleton* Singleton::getInstance()
{
if(instance == NULL)
instance = new Singleton();
return instance;
}
int main()
{
cout<<"begin main\n";
Singleton* instance = Singleton::getInstance();
cout<<"end main\n";
}
但是這樣構(gòu)造函數(shù)不會自動調(diào)用,需要用戶這個類的用戶手動delete instance. 這樣是不太好的
這是懶漢式的,在多線程的情況下需要同步。也可以寫成餓漢式的,但是c++里面寫成餓漢式的不太好,如果有多個單例類,而他們是相互引用的,那么餓漢式就有可能出問題。因為在C++中,這幾個單例類的靜態(tài)成員的初始化順序是不確定的。
二、2B的Singleton
#include<iostream>
using namespace std;
class Singleton
{
public:
static Singleton* getInstance();
private:
static Singleton* instance;
Singleton()
{
cout<<"constructor\n";
// do something
};
~Singleton()
{
cout<<"destructor\n";
//do something
}
class Garbo
{
public:
~Garbo()
{
cout<<"Garbo destructor\n";
if(Singleton::instance != NULL)
delete Singleton::instance;
}
};
static Garbo garbo;
};
Singleton* Singleton::instance = NULL;
Singleton::Garbo Singleton::garbo;
Singleton* Singleton::getInstance()
{
if(instance == NULL)
instance = new Singleton();
return instance;
}
int main()
{
cout<<"begin main\n";
Singleton* instance = Singleton::getInstance();
cout<<"end main\n";
}
用了一個內(nèi)部類, Garbo。 由于main結(jié)束后,會自動釋放 garbo, 而garbo就會調(diào)用instance的destructor.
三、文藝Singleton
#include<iostream>
using namespace std;
class Singleton
{
public:
static Singleton* getInstance();
private:
static Singleton instance;
Singleton()
{
cout<<"constructor\n";
// do something
};
~Singleton()
{
cout<<"destructor\n";
//do something
}
};
Singleton Singleton::instance;
Singleton* Singleton::getInstance()
{
return &instance;
}
int main()
{
cout<<"begin main\n";
Singleton* instance = Singleton::getInstance();
cout<<"end main\n";
}
instance在main開始前就已經(jīng)構(gòu)造好了,在main結(jié)束后會自動釋放。
但是這樣的話,只能是餓漢式的singleton。如果在構(gòu)造函數(shù)中需要申請大量資源,這些資源就一直存在在整個運行階段。而不管這些資源什么時候需要,什么時候不許要。
四、改進的文藝Singleton
#include<iostream>
using namespace std;
class Singleton
{
public:
static Singleton* getInstance();
private:
Singleton()
{
cout<<"constructor\n";
// do something
};
~Singleton()
{
cout<<"destructor\n";
//do something
}
};
Singleton* Singleton::getInstance()
{
static Singleton instance;
return &instance;
}
int main()
{
cout<<"begin main\n";
Singleton* instance = Singleton::getInstance();
cout<<"end main\n";
}
這樣,就可以在需要的時候再構(gòu)造singleton, 也就是懶漢式的。
相關(guān)文章
深入探討:main函數(shù)執(zhí)行完畢后,是否可能會再執(zhí)行一段代碼?
本篇文章是對main函數(shù)執(zhí)行完畢后,是否可能會再執(zhí)行一段代碼,進行了詳細的分析介紹,需要的朋友參考下2013-05-05c語言 數(shù)據(jù)結(jié)構(gòu)實現(xiàn)之字符串
這篇文章主要介紹了c語言 數(shù)據(jù)結(jié)構(gòu)實現(xiàn)之字符串的相關(guān)資料,需要的朋友可以參考下2017-05-05VS報錯C6011的問題:取消對NULL指針的引用(解決方法)
這篇文章主要介紹了VS報錯C6011的問題:取消對NULL指針的引用(解決方法),C6011:取消對NULL指針的引用,發(fā)現(xiàn)是沒有進行空指針的判斷,解決方案跟隨小編一起看看吧2024-01-01深入理解C++的動態(tài)綁定與靜態(tài)綁定的應(yīng)用詳解
本篇文章是對C++中的動態(tài)綁定與靜態(tài)綁定進行了詳細的分析介紹,需要的朋友參考下2013-05-05