目录
[1. 单例模式介绍](#1. 单例模式介绍)
1. 单例模式介绍
有些时候我们在做 qt 项目的时候,要用到很多类. 例如我们用到的类有 A,B,C,D. 其中,A 是 B,C,D 中都需要用到的类,A 类非常的抢手. 但是,A 类非常的占内存,定义一个 A 对象需要 500M 内存,假如在 B,C,D 中都定义一个 A 类对象,对 内存的消耗是可想而知的.
所以 B,C,D 分别都定义一个 A 类对象是不可能的. 那么我们此时就希望: 能不能把 A 定义成"全局变量",然后这样子 B,C,D 类都能访问,并且整个程序就只有这一个 A?
答案是可以的,定义 A 的时候以单例模式定义即可
单例模式作为一种常用的软件设计模式,主要是用来保证系统中只有一个实例,例如一般一个程序中只有一 个日志输出实例,一个系统中只有一个数据库连接实例,这时候用单例模式非常合适。
单例模式用来做什么
正如之前所说,单例模式:
(1) 整个程序只有一个对象;
(2) 整个程序都能访问到它;
(3) 分为"懒汉模式"和"饿汉模式";
(4) "懒汉模式"是用到单例的时候才创建,否则不创建对象;
(5) "饿汉模式"是在程序启动时就需要创建变量;
懒汉模式是时间换空间,饿汉模式是空间换时间。
单例模式是在大型项目用到非常多的,例如:
(1) 一个日记记录类,必须整个程序只能有一个,假如有 2 个,在写日志文件时会相互"践踏".
(2) 数据库操作,连接类,必须整个程序只能有一个,假如有 2 个就会出问题,你想看假如 2 个进程同时使用 同一个账号,然后写入同一个数据会发生什么事情...
单例模式有效的解决了重复定义对象的问题,假如配合互斥锁,还可以解决多进程,线程之间的同步互斥问题, 也就是同一时间只能有一个进程操作单例对象。
2.单例模式实现
设计单例模式非常简单:
cpp文件实例:
cpp
//这句一定要写,因为 self 是静态变量,必须要初始化值
ClassA* ClassA::self = NULL;
ClassA *ClassA::getInstance()
{
//判断是否是第一次调用
//假如不为 NULL,会直接 return
if (self == NULL) {
//防止多进程同时调用创建 2 个对象的问题
//原理是因为是静态变量只会定义一次
static QMutex mutex;
//上锁
QMutexLocker locker(&mutex);
//在判断一次,防止其它进程抢先 new 过了
if (self == NULL) {
self = new ClassA;
}
}
return self; //返回指针
}
h头文件实例:
cpp
class ClassA : public QObject
{
Q_OBJECT
public:
//通过它获取 self 指针,这个函数是静态调用,不需要创建对象就可以访问
static ClassA *getInstance(); //返回自身的指针
//构造函数
explicit ClassA(QObject *parent = 0);
void hello(){ //一个普通函数
qDebug() << "hello world";
}
private:
static ClassA *self; //静态指针
};
#endif // ICONFONT_H
在其他类里的调用方法:
ClassA::getInstance()->hello();
其中:
ClassA::getInstance()会获取全局的 self 指针,通过这个指针我们可以访问到里边的函数。在任何类里,都可以 无条件访问 A 对象,并且不需要创建 A 对象,而且 A 对象全局只有一个实例:
(1) 在类 A 里定义一个静态的指针 self;
(2) 当调用获取 A 对象的指针时,判断这个 self 是否为空,假如为空则创建新对象并赋值给 self 并返回 self,存 在则直接返回 self;
在此使用用的最多的"懒汉模式"进行演示:
cpp 代码如下(以数据库为例(多个 ui 界面都使用到此类)):
cpp
#include "qmqttclient.h"
#include <QMutex>
#include <QMutexLocker>
#include <QMessageBox>
//构造函数
QmqttClient::QmqttClient()
{
}
//这句一定要写,因为 self 是静态变量,必须要初始化值
QmqttClient* QmqttClient::mqttSelf = NULL;
QmqttClient *QmqttClient::getInstance()
{
//判断是否是第一次调用
//假如不为 NULL,会直接 return
if (mqttSelf == NULL) {
//防止多进程同时调用创建 2 个对象的问题
//原理是因为是静态变量只会定义一次
static QMutex mutex;
//上锁 QMutexLocker 会锁住 mutex,当 QMutexLocker 被释放的时候自动解锁
//locker 是局部变量,所以 getInstance 函数结束时会自动解锁
QMutexLocker locker(&mutex);
//在判断一次,防止其它进程抢先 new 过了
if (mqttSelf == NULL) {
mqttSelf = new QmqttClient;
}
}
return mqttSelf; //返回指针
}
头文件
cpp
#ifndef QMQTTCLIENT_H
#define QMQTTCLIENT_H
#include <QObject>
#include "qmqtt.h"
#include <QtCore>
class QmqttClient : public QObject{
Q_OBJECT
public:
//通过它获取 self 指针,这个函数是静态调用,不需要创建对像就可以访问,函数名
自定义
//这个函数的作用是给别人获取它的静态对象 返回自身的指针
static QmqttClient *getInstance();
// void hello(){ //一个普通函数
// qDebug() << "hello world";
private:
QmqttClient(); //构造函数,写在 private 下,不允许 new 生成单例
static QmqttClient *mqttSelf;//静态指针
};
在其他类里的调用方法:
QmqttClient::getInstance()->mqttClient->setHost(ui->lEditServerIP->text()); //服务 器 IP QmqttClient::getInstance()->mqttClient->setPort(1883);//端口号
其中: QmqttClient::getInstance()会获取全局的 self 指针,通过这个指针我们可以访问到里边的函数. 在任何类里,都可以无条件访问 A 对象,并且不需要创建 A 对象,而且 A 对象全局只有一个实例: 注意:可以把多个对象放到单例模式