单例模式是一种创建型设计模式,它确保一个类只有一个实例,并提供一个全局访问点来访问这个唯一实例。单例模式通常用于管理全局资源,如配置、数据库连接等。
单例模式有两种初始化策略,一种饿汉式,一种懒汉式
饿汉式
这种方式在类加载时就完成了实例的初始化。因为类加载是由JVM的类加载器负责的,并且JVM的类加载器是线程安全的,所以这种方式是线程安全的,无需额外的同步措施。
public class Singleton {
private static Singleton instance = new Singleton();
private Singleton() {}
public static Singleton getInstance() {
return instance;
}
}
懒汉式
这种方式在真正需要实例的时候(比如调用getInstance方法时)才进行初始化。这种方式在并发环境下可能会产生多个实例,因为可能同时有多个线程进入getInstance方法并且看到instance为null,所以通常会加上同步措施来保证线程安全。
以下是不安全没有同步的例子:
public class Singleton {
private static Singleton instance;
private Singleton() {}
public static Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
这个线程不安全的懒汉式版本在多线程环境下可能会导致多个实例被创建,因为它没有同步措施来确保instance的赋值操作是原子的。
饿汉式的延迟初始化
这是使用双重检查锁定(Double-Checked Locking)的懒汉式实现
public class Singleton {
private volatile static Singleton instance;
private Singleton() {}
public static Singleton getInstance() {
if (instance == null) {
synchronized (Singleton.class) {
if (instance == null) {
instance = new Singleton();
}
}
}
return instance;
}
}
这里的volatile关键字,它用于确保在多线程环境下instance的可见性。同时,双重检查锁定也减少了不必要的同步开销。
实现:简单实现懒汉式单例模型
1.首先,确保dynadbmag类有一个私有的静态指针来存储它的唯一实例
private:
Ui::dynadbmag *ui;
static dynadbmag * pInstance;
2.初始化静态成员
dynadbmag * dynadbmag::pInstance = NULL;
3.使用GetInstance()方法获取静态实例
public:
explicit dynadbmag(QWidget *parent = nullptr);
static dynadbmag* GetInstance();
dynadbmag * dynadbmag::GetInstance()
{
if (!pInstance)
{
pInstance = new dynadbmag;
}
return pInstance;
}