【C/C++】构造函数被私有化,为什么还可以在类外直接new一个对象?

实例代码:

c 复制代码
// 饿汉模式 -> 定义类的时候创建单例对象
// 定义一个单例模式的任务队列
class TaskQueue
{
public:
    TaskQueue(const TaskQueue & t) = delete;
    TaskQueue& operator=(const TaskQueue& t) = delete;
    static TaskQueue* getInstance()
    {
        return m_taskQ;
    }

    void print()
    {
        cout << "我是单例对象的一个成员函数..." << endl;
    }

private:
    TaskQueue() = default;
    // TaskQueue(const TaskQueue& t) = default;
    // TaskQueue& operator=(const TaskQueue& t) = default;
    // 只能通过类名访问静态属性或方法
    static TaskQueue* m_taskQ;
};

TaskQueue* TaskQueue::m_taskQ = new TaskQueue;		// 加了类名作用域,实质仍在类内。

构造函数私有化(private),构造函数就不能通过外部访问,构造函数不能访问,自然就不能通过new创建一个新对象(new对象时,一定会初始化,即调用构造函数)。

那这里为什么还能 new TaskQueue 呢?还可以在类外直接new一个对象?

c 复制代码
TaskQueue* TaskQueue::m_taskQ = new TaskQueue;		// 返回new出来对象的地址

因为私有是相对作用域外,同一作用域没有这些概念,都是透明的。上面代码添加了类名作用域(TaskQueue::),表示这一句代码仍在类内,只不过是语法要求写在外面(在C++中,类的静态成员(static member)必须在类内声明,在类外初始化)。静态成员在外部初始化是语法要求,但初始化实质算在作用域内操作。

单例模式完整实例代码:

c 复制代码
#include<iostream>
using namespace std;


// 饿汉模式 -> 定义类的时候创建单例对象
// 定义一个单例模式的任务队列
class TaskQueue
{
public:
	TaskQueue(const TaskQueue & t) = delete;
	TaskQueue& operator=(const TaskQueue& t) = delete;
	static TaskQueue* getInstance()
	{
		return m_taskQ;
	}

	void print()
	{
		cout << "我是单例对象的一个成员函数..." << endl;
	}

private:
	TaskQueue() = default;
	// TaskQueue(const TaskQueue& t) = default;
	// TaskQueue& operator=(const TaskQueue& t) = default;
	// 只能通过类名访问静态属性或方法
	static TaskQueue* m_taskQ;
};

TaskQueue* TaskQueue::m_taskQ = new TaskQueue;		// 加了类名作用域,实质仍在类内。


int main()
{
	TaskQueue* m_taskQ = TaskQueue:: getInstance();
	m_taskQ->print();

	return 0;
}

m_taskQ 是私有的(private),类外不能访问,只能通过公用的成员函数(public)访问。能够操纵静态成员变量的函数,只有静态成员函数。

作用:在单例设计中,Singleton类在定义的时候将构造方法私有化,而在内部仅仅new出一个对象,是为了禁止别的类在外面直接new Singleton()出来,这样你只要在Singleton类中new一个对象,就能确保无论什么情况都只会产生一个Singleton对象,外部无法new出来,内部已经定义好了,有且仅有一个对象,这就是单列设计的核心。;单例设计模式目标是保证一个类在内存中对象的唯一性。所以要保证自己的类中创建了对象之后,其他的类不能再创建对象,只能获取这个对象。所以只要将本类中的构造函数私有化,其他程序就无法再创建该类对象。

相关推荐
咩咦21 小时前
C++学习笔记20:日期类比较运算符重载
c++·学习笔记·类和对象·运算符重载·比较运算符·日期类
paeamecium21 小时前
【PAT甲级真题】- A+B in Hogwarts
c++·算法·pat考试·pat
咩咦21 小时前
C++学习笔记16:构造函数
c++·学习笔记·类和对象·构造函数·默认构造函数
basketball6161 天前
C++ 嵌套类完全指南:类中类的巧妙设计
开发语言·c++
kyle~1 天前
ros_gz_bridge---底层通信的实现
c++·机器人·仿真·ros2
Jasmine_llq1 天前
《B4261 [GESP202503 三级] 2025》
开发语言·c++·算法·条件判断算法·位运算恒等式推导·简单算术运算
小张成长计划..1 天前
【C++】32:智能指针
c++
咩咦1 天前
C++学习笔记19:运算符重载基础与赋值运算符重载
c++·学习笔记·类和对象·运算符重载·赋值运算符·operator
无限进步_1 天前
C++异常机制:抛出、捕获与栈展开
开发语言·c++·安全
王老师青少年编程1 天前
csp信奥赛C++高频考点专项训练之前缀和&差分 --【一维前缀和】:宝石串
c++·前缀和·csp·高频考点·信奥赛·宝石串