在我们的学习中不免会遇到一些要设计一些特殊的类,要求这些类只能在内存中特定的位置创建对象,这就需要我们对类进行一些特殊的处理,那我们该如何解决呢?下面来絮叨絮叨~~
拷贝只会放生在两个场景中:
具体代码实现:
class CopyBan
{
// ...
private:
CopyBan(const CopyBan&);
CopyBan& operator=(const CopyBan&);
};
class CopyBan
{
// ...
CopyBan(const CopyBan&) = delete;
CopyBan& operator=(const CopyBan&) = delete;
// ...
};
分析题目既然只能将对象创建在堆上,那么我们就需要将在栈上或数据段
创建对象的这种情况给屏蔽掉。
首先我们我们的思路:
下面我们要做的就是要单独开道口子给在堆上创建对象用~~
虽然我们在类外用不了构造函数,但是我们可以在类内调用,所以我们在类内调用一下构造函数,并将创建的对象返回出来。
class HeapOnly
{
public:
static HeapOnly* CreateObj()
{
//这里new直接去调用构造函数去了,类里面直接调用没限制
return new HeapOnly;
}
//防拷贝,不让拷贝(形参可写可不写)
HeapOnly(const HeapOnly&) = delete;
//赋值并不会影响创建的对象跑到其他的地方去了,赋值不是创建对象的
//拷贝构造反而是用来创建对象的
private:
//构造函数私有 -- 将构造函数封起来(三条路都封死)
HeapOnly()
{}
};
问题:
CreateObj()
来创建对象的时候有个很尬的问题CreateObj()
函数设成静态函数补充:
此时还有个问题就是,拷贝构造也是会在栈上创建对象,我们可以将拷贝构造设成私有,或者C++11中直接将拷贝构造给删除掉。
通过类域直接调用构造函数:
int main()
{
//HeapOnly h1;//栈
//static HeapOnly h2;//静态区
//HeapOnly* ph3 = new HeapOnly;//堆
//此时无论如何创建的对象都是在堆上的
HeapOnly* ph4 = HeapOnly::CreateObj();
HeapOnly* ph5 = HeapOnly::CreateObj();
//创建在堆上,但是拷贝还是拷贝在了栈上
//HeapOnly copy(*ph4);
delete ph4;
delete ph5;
return 0;
}
相比于上一种方法(将构造函数和拷贝构造私有或者删除),方法二显得更加牵强一点,将析构函数设成私有的,这样当对象生命周期结束时自动调用析构函数时会调用不到。
首先我们我们的思路:
解决办法:类外调用不了私有的析构函数,但是类内可以调用,所以我们可以提供一个接口
class HeapOnly
{
public:
static void DelObj(HeapOnly* ptr)
{
delete ptr;
}
//比较花的玩法 -- 这样还不用传参了
/*void DelObj()
{
delete this;
}*/
private:
//析构函数私有
~HeapOnly()
{}
};
int main()
{
//HeapOnly h1;
//static HeapOnly h2;
HeapOnly* ph3 = new HeapOnly;
//delete ph3;
//方法一:要传参
ph3->DelObj(ph3);
//方法二:不用传参
//ph3->DelObj();
return 0;
}
还有个比较花的玩法,直接在类内将this
指针释放掉,这种写法也没问题,就是很少见。
类似于上一个问题中的方法,我们先将三条路封死,然后单独给在栈上创建对象开条出路。
首先我们我们的思路:
class StackOnly
{
public:
static StackOnly CreateObj()
{
//局部对象,不能用指针返回,也不能用引用返回,只能传值返回
//传值返回必然会有一个拷贝构造发生
return StackOnly();
}
void Print()
{
cout << "Stack Only" << endl;
}
private:
//构造函数私有
StackOnly()
{}
};
int main()
{
StackOnly h1 = StackOnly::CreateObj();
//不用对象去接受
StackOnly::CreateObj().Print();
//static StackOnly h2;
//禁掉下面的玩法
//StackOnly* ph3 = new StackOnly;
return 0;
}
和上一个题不同的地方,这里不能将拷贝构造给禁掉,因为CreateObj()
是传值返回,传值返回必然会有一个拷贝构造(不考虑编译器优化的问题)。
再来个比较花的玩法,我们要禁掉在堆上创建对象的情况,所以我们可以对new
下手。
之前我们学习C++内存管理时,我们知道new一共分为两个步骤:开空间 + 调用构造函数。
new
在底层会调用operator new
new、new[]、delete 和 delete[]
也可以进行重载补充:
- 类专属的operator new函数属于new操作符的重载。它提供了一种自定义内存分配策略的方法,用于为该类的对象分配内存空间。而全局的operator new函数则为所有类型的对象提供默认的内存分配策略。
- 编译器会在类中查找是否有声明或定义了类专属的operator new函数。如果找到了,则编译器会使用该函数来分配对象的内存空间。如果没有找到,编译器会继续往父类查找是否存在类专属的operator new函数,直到找到一个已经声明或定义了该函数的类或者到达继承体系的根部为止。
- 即使某个类没有定义类专属的operator new函数,编译器仍然可以使用全局的operator new函数进行内存分配。因此,类专属的operator new函数并不是必需的,但它可以提供一种更加灵活和精细的内存分配策略,以满足特定的需求。
在具备上述知识体系之后,我们在类内实现这个类专属的operator new
,并且我们直接将其删掉,这样在堆上创建对象这一个操作就执行不了了。
class StackOnly
{
public:
//new由两部分构成,一部分调用operator new,另一部分是调用构造函数
//直接把operator new给禁掉,不禁构造函数
//类内专属重载了,一个类的专属的operator new,并且将其删掉
void* operator new(size_t size) = delete;
void operator delete(void* p) = delete;
};
//全局的也禁不掉
StackOnly s3;
int main()
{
//new会自动调用operator new
//这里的new就不会去调用全局的,而是调用类里的
//此时类内的operator new 被删掉了,就调用不到了
//StackOnly* p1 = new StackOnly;
//StackOnly s1;
//静态的和全局的禁不掉
static StackOnly s2;
return 0;
}
不过还是有的情况防不住,那就是在数据段(静态对象和全局对象)禁不掉。
这个内容我们在继承那一节中详细讲过,不熟悉的小伙办可以回去复习哦, 传送门
概念:
一个类只能创建一个对象,即单例模式,该模式可以保证系统中该类只有一个实例,一个进程只能有一个对象,并提供一个访问它的全局访问点,该实例被所有程序模块共享。
class Singleton
{
public:
static Singleton* GetInstance()
{
cout << _spInst << endl;
return _spInst;
}
private:
Singleton()
{}
//防不住拷贝,直接将其删除
Singleton(const Singleton&) = delete;
static Singleton _sInst; //声明
static Singleton* _spInst; //声明
//饿汉模式初始化一个成员
int _a = 0;
};
Singleton Singleton::_sInst; //定义
Singleton* Singleton::_spInst = new Singleton; //定义
int main()
{
//GetInstance() 可以或者这个Singleton类的单例对象
Singleton::GetInstance()->Print();
//在外面就定义不出来对象了
//Singleton st1;
//Singleton* st2 = new Singleton;
//拷贝删除掉
//Singleton copy(*Singleton::GetInstance());
return 0;
}
思路:
main
函数之前)就创建好了main
函数之前就已经创建好对象了main
函数之前创建的我们将拷贝构造给禁掉了:
注意:
饿汉,对象在main函数之前就创建了,导致了程序启动起来很慢。
如果单例对象构造十分耗时或者占用很多资源,比如加载插件, 初始化网络连接,读取文件等等,而有可能该对象程序运行时不会用到,那么也要在程序一开始就进行初始化,就会导致程序启动时非常的缓慢。 所以这种情况使用懒汉模式(延迟加载)更好。
class InfoMgr
{
public:
static InfoMgr* GetInstance()
{
//还需要加锁,这个后面讲 -- 双检查加锁
if (_spInst == nullptr)
{
_spInst = new InfoMgr;
}
return _spInst;
}
void SetAddress(const string& s)
{
_address = s;
}
string& GetAddress()
{
return _address;
}
//实现一个内嵌垃圾回收类
class CGarbo
{
public:
~CGarbo()
{
if (_spInst)
{
delete _spInst;
}
}
};
//定义一个静态成员变量,程序结束时,系统会自动调用它的析构函数从而释放单例对象
static CGarbo Garbo;
private:
InfoMgr()
{}
~InfoMgr()
{
//假设析构时需要信息写到文件持久化,目的是单利对象销毁时要做一些必不可少的动作
}
InfoMgr(const InfoMgr&) = delete;
string _address;
int _secretKey;
static InfoMgr* _spInst; //声明
};
//先初始化成空
InfoMgr* InfoMgr::_spInst = nullptr; //定义
InfoMgr::CGarbo Garbo;
int main()
{
//全局只有一个InfoMgr对象
InfoMgr::GetInstance()->SetAddress("122333");
cout << InfoMgr::GetInstance()->GetAddress() << endl;
return 0;
}
域饿汉模式不同的是,懒汉是模式则是一开始不创建对象,而是一开始先给一个指针,这样就避免了程序启动起来很慢的问题。
思路:
补充:
虽然最终申请的单例对象还是会还给操作系统,但是我们还是手动释放一下更好~