Java单例模式分析

作者:jaywangpku 时间:2023-11-16 03:17:06 

单例模式

为什么要用单例

确保某个类只有一个对象,常用于访问数据库操作,服务的配置文件等。

单例的关键点

1、默认构造函数为private,复制构造函数和复制赋值函数也要private或=delete禁用。(做到无法被外部其他对象构造)

2、通过一个静态方法或枚举返回单例类对象。

3、确保多线程的环境下,单例类对象只有一个。

几种写法

本文主要介绍C++的懒汉式和饿汉式写法。

懒汉式

需要生成唯一对象时(调用GetInstance时),才生成

线程不安全的错误写法


class SingleInstance
{
public:
   // 静态方法获取单例
   static SingleInstance *GetInstance();
   // 释放单例避免内存泄露
   static void deleteInstance();
private:
   SingleInstance() {}
   ~SingleInstance() {}
   // 复制构造函数和复制赋值函数设置为private,被禁用
   SingleInstance(const SingleInstance &signal);
   const SingleInstance &operator=(const SingleInstance &signal);
private:
   static SingleInstance *m_SingleInstance;
};

// 初始化为NULL,与后面形成对比
SingleInstance *SingleInstance::m_SingleInstance = NULL;

SingleInstance* SingleInstance::GetInstance()
{
   // 多线程情况下,一个线程通过if检查但是还未new出单例时,另一个线程也通过了if检查,导致new出多个对象
   if (m_SingleInstance == NULL)
   {
       m_SingleInstance = new (std::nothrow) SingleInstance;
   }
   return m_SingleInstance;
}

void SingleInstance::deleteInstance()
{
   if (m_SingleInstance)
   {
       delete m_SingleInstance;
       m_SingleInstance = NULL;
   }
}

线程安全的双检锁写法


class SingleInstance
{
public:
   // 静态方法获取单例
   static SingleInstance *GetInstance();
   // 释放单例避免内存泄露
   static void deleteInstance();
private:
   SingleInstance() {}
   ~SingleInstance() {}
   // 复制构造函数和复制赋值函数设置为private,被禁用
   SingleInstance(const SingleInstance &signal);
   const SingleInstance &operator=(const SingleInstance &signal);
private:
   static SingleInstance *m_SingleInstance;
};

// 初始化为NULL,与后面形成对比
SingleInstance *SingleInstance::m_SingleInstance = NULL;

SingleInstance* SingleInstance::GetInstance()
{
   // 如果直接在外面锁,功能也ok,但每次运行到这个地方便需要加一次锁,非常浪费资源
   // 在里面加锁,初始化时存在加锁的情况,初始化之后,外层if都为false,直接返回,避免加锁
   if (m_SingleInstance == NULL)
   {
       std::unique_lock<std::mutex> lock(m_Mutex); // Lock up
       if (m_SingleInstance == NULL)
       {
           m_SingleInstance = new (std::nothrow) SingleInstance;
       }
   }
   return m_SingleInstance;
}

void SingleInstance::deleteInstance()
{
   if (m_SingleInstance)
   {
       delete m_SingleInstance;
       m_SingleInstance = NULL;
   }
}

线程安全的局部静态变量写法

推荐


class SingleInstance
{
public:
   // 静态方法获取单例
   static SingleInstance *GetInstance();
private:
   SingleInstance() {}
   ~SingleInstance() {}
   // 复制构造函数和复制赋值函数设置为private,被禁用
   SingleInstance(const SingleInstance &signal);
   const SingleInstance &operator=(const SingleInstance &signal);
};

SingleInstance& SingleInstance::GetInstance()
{
// 局部静态变量(一般为函数内的静态变量)在第一次使用时分配内存并初始化。
   static SingleInstance m_SingleInstance;
   return m_SingleInstance;
}

饿汉式

进程运行前(main函数执行),就创建

线程安全的进程运行前初始化写法


class SingleInstance
{
public:
   // 静态方法获取单例
   static SingleInstance *GetInstance();
private:
   SingleInstance() {}
   ~SingleInstance() {}
   // 复制构造函数和复制赋值函数设置为private,被禁用
   SingleInstance(const SingleInstance &signal);
   const SingleInstance &operator=(const SingleInstance &signal);
private:
static SingleInstance *m_SingleInstance;
};

SingleInstance* SingleInstance::GetInstance()
{
   return m_SingleInstance;
}

// 全局变量、文件域的静态变量和类的静态成员变量在main执行之前的静态初始化过程中分配内存并初始化
Singleton* Singleton::g_pSingleton = new (std::nothrow) Singleton;
int main()
{
return 0;
}

线程安全的类静态成员变量写法


class SingleInstance
{
public:
   // 静态方法获取单例
   static SingleInstance *GetInstance();
   // 全局变量、文件域的静态变量和类的静态成员变量在main执行之前的静态初始化过程中分配内存并初始化。
   static SingleInstance m_SingleInstance;
private:
   SingleInstance() {}
   ~SingleInstance() {}
   // 复制构造函数和复制赋值函数设置为private,被禁用
   SingleInstance(const SingleInstance &signal);
   const SingleInstance &operator=(const SingleInstance &signal);
};

SingleInstance& SingleInstance::GetInstance()
{
   return m_SingleInstance;
}

静态内部类写法

JAVA


/**
* 静态内部类实现单例模式
*/
public class Singleton {
   private Singleton() {
   }

public static Singleton getInstance() {
       return SingletonHolder.instance;
   }

/**
    * 静态内部类
    */
   private static class SingletonHolder {
       private static Singleton instance = new Singleton();
   }
}

第一次加载Singleton类时不会初始化instance,只有在第一次调用getInstance()方法时,虚拟机会加载SingletonHolder类,初始化instance。
这种方式既保证线程安全,单例对象的唯一,也延迟了单例的初始化,推荐使用这种方式来实现单例模式。

枚举单例

JAVA


/**
* 枚举实现单例模式
*/
public enum SingletonEnum {
   INSTANCE;
   public void doSomething() {
       System.out.println("do something");
   }
}

默认枚举实例的创建是线程安全的,即使反序列化也不会生成新的实例,任何情况下都是一个单例。
优点: 简单!

容器实现单例

JAVA


import java.util.HashMap;
import java.util.Map;
/**
* 容器类实现单例模式
*/
public class SingletonManager {
   private static Map<String, Object> objMap = new HashMap<String, Object>();

public static void regsiterService(String key, Object instance) {
       if (!objMap.containsKey(key)) {
           objMap.put(key, instance);
       }
   }

public static Object getService(String key) {
       return objMap.get(key);
   }
}

SingletonManager可以管理多个单例类型,使用时根据key获取对象对应类型的对象。这种方式可以通过统一的接口获取操作,隐藏了具体实现,降低了耦合度。

参考

单例模式的6种实现方式

软件开发常用设计模式—单例模式总结(c++版)

https://stackoverflow.com/questions/1008019/c-singleton-design-pattern

https://programmer.ink/think/summary-of-c-thread-safety-singleton-patterns.html

来源:https://blog.csdn.net/u013095333/article/details/120537938

标签:Java,单例模式,分析
0
投稿

猜你喜欢

  • Spring Boot ActiveMQ连接池配置过程解析

    2023-11-08 23:08:02
  • springboot项目打成war包部署到tomcat遇到的一些问题

    2023-10-12 12:46:46
  • Netty 拆包沾包问题解决方案详解

    2023-09-01 21:58:50
  • springcloud干货之服务注册与发现(Eureka)

    2021-07-08 11:04:23
  • Android Jetpack架构组件Lifecycle详解

    2023-03-04 22:21:01
  • Spring boot整合log4j2过程解析

    2023-11-29 10:47:53
  • 一文详解Java Netty中的Constant类

    2023-03-03 17:17:55
  • c#3.0实现延迟赋值示例

    2023-01-04 16:28:51
  • 阿里开源Java诊断工具神器使用及场景详解

    2023-11-06 17:24:21
  • 修改idea的这些启动参数,令你的idea健步如飞

    2021-11-14 14:48:11
  • springboot整合Quartz实现动态配置定时任务的方法

    2023-03-08 22:13:10
  • java基于JDBC连接Oracle 11g Release2实例分析

    2022-06-06 02:36:36
  • rsa加密算法使用示例分享

    2021-08-03 20:47:29
  • Java原生HttpClient的使用详解

    2022-06-04 16:29:49
  • 浅谈Java finally语句到底是在return之前还是之后执行(必看篇)

    2022-10-20 06:58:41
  • SpringAOP切点函数实现原理详解

    2021-09-05 11:00:59
  • ViewPager顶部导航栏联动效果(标题栏条目多)

    2022-11-03 13:19:38
  • 深入理解Spring中bean的生命周期介绍

    2023-02-08 17:21:37
  • java web实现分页查询实例方法

    2022-12-18 18:07:57
  • C#判断一个字符串是否是数字或者含有某个数字的方法

    2022-05-07 14:38:17
  • asp之家 软件编程 m.aspxhome.com