设计模式之单例模式

本文深入解析单例模式,一种常用的设计模式,旨在确保系统中一个类只有一个实例,并提供全局访问点。文章详细介绍了单例模式的实现原理,包括如何通过私有构造函数和静态方法控制对象的创建和访问,以及如何避免内存泄漏。
摘要由CSDN通过智能技术生成

以下是我摘自维基百科,我觉得总结的特别好: 

单例模式,也叫单子模式,是一种常用的软件设计模式,属于创建型模式的一种。在应用这个模式时,单例对象的类必须保证只有一个实例存在。许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为。比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息。这种方式简化了在复杂环境下的配置管理。

实现单例模式的思路是:一个类能返回对象一个引用(永远是同一个)和一个获得该实例的方法(必须是静态方法,通常使用getInstance这个名称);当我们调用这个方法时,如果类持有的引用不为空就返回这个引用,如果类保持的引用为空就创建该类的实例并将实例的引用赋予该类保持的引用;同时我们还将该类的构造函数定义为私有方法,这样其他处的代码就无法通过调用该类的构造函数来实例化该类的对象,只有通过该类提供的静态方法来得到该类的唯一实例。

现在让我们结合代码看看单例模式的思想:

class Weight {
public:
	static Weight* GetInstance() {    //访问该类唯一实例的全局访问点(静态访问点)
		if (weight == nullptr) {      //当weight指针为nullptr时我们就创建该类的唯一实例(构造函数为私有的),不为nullptr时,我们便可以调用该类的方法
			weight = new Weight();
		}
		return weight;
	}

private:
	Weight() { std::cout << "构造" << std::endl; };
	~Weight(){
		if (weight) {
			delete weight;
		}
		std::cout << "析构" << std::endl;
	}
	static Weight* weight;  //静态数据成员,由该类进行维护
};
Weight* Weight::weight = nullptr;


以上代码虽然已经很完美了,但是却有一个很大的问题:weight的析构函数并没有被执行,这样很可能会造成内存泄漏。那怎么办呢?我们可以手动的释放weight指针所指的内存。

#include<iostream>

class Weight {
public:
	static Weight* GetInstance() {    //访问该类唯一实例的全局访问点(静态访问点)
		if (weight == nullptr) {      //当weight指针为nullptr时我们就创建该类的唯一实例(构造函数为私有的),不为nullptr时,我们便可以调用该类的方法
			weight = new Weight();
		}
		return weight;
	}
	void Delete() {
		if (weight != nullptr) {
			delete weight;
			std::cout << "析构" << std::endl;
			
		}
	}

private:
	Weight() { std::cout << "构造" << std::endl; };
	static Weight* weight;  //静态数据成员,由该类进行维护
};
Weight* Weight::weight = nullptr;


但是这样显得很麻烦,而且如果我们忘记手动释放内存的话,也可能会造成内存泄漏。那怎么办呢?我们知道,进程在执行前会进行进程执行环境的初始化操作,其中包括对全局变量或者静态变量的初始化。当进程执行完后,会对进程所占有的内存进行收回,当然啦,也会对全局变量或者静态变量进行析构。

#include<iostream>

class Weight {
public:
	static Weight* GetInstance() {
		if (weight == nullptr) {
			weight = new Weight();
			ds =Weight::DeleteWeight();
		}

		return weight;
	}
private:
	struct DeleteWeight {
		DeleteWeight() {};
		~DeleteWeight() {
			if (weight != nullptr) {
				delete weight;
				std::cout << "析构" << std::endl;
			}
		}
	};
	Weight() {
		std::cout << "构造" << std::endl;
	}

private:
	static Weight* weight;
	static DeleteWeight ds;
};

Weight* Weight::weight = nullptr;

如上代码所示,我们进行嵌套类设计,然后让该类为静态私有的,这样我们便可以让类自己在合适的时候对类的唯一实例进行析构。

以上便是我对单例模式的理解,如有错误请告知,谢谢啦~

 

参考:https://blog.csdn.net/Hackbuteer1/article/details/7460019

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值