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


猜你喜欢
- 背景公司线上有个tomcat服务,里面合并部署了大概8个微服务,之所以没有像其他微服务那样单独部署,其目的是为了节约服务器资源,况且这8个服
- JPA Specification常用查询+排序1.第一步:继承父类public interface TblCarton2RCardLogR
- MyBatis一对多的xml配置用的是window上面的画图板,没法以文字的方式展示出来,见谅嵌套查询嵌套结果一对多关联查询xml配置写法
- 本文主要介绍android应用android系统中剪切板进行数据的传递,首先讲解的是传递简单数据,然后讲解传递对象类型的数据。所有实例均在a
- SpringBoot启动类静态资源路径SpringBoot核心配置类SpringBoot核心JAR包--》spring-boot-autoc
- 普通Java-Kotlin类添加注释添加类时注释作者信息和日期时间依次打开File—>Settings—>editor—>
- 本文研究的主要是Java编程利用openoffice将doc、docx转为pdf的实现代码,具体如下。1. 需要用的软件OpenOffice
- 目录RemoveSubstringReplaceSpiltJoinAppendRemoveRemove(int startIndex) 删除
- 如下所示:Ctrl+1或F2快速修复Ctrl+D快捷删除行Shift+Enter 快速切换到下一行,在本行的任何位置都可Ctrl+F11快速
- 最近比较空闲没有项目做,于是乎捋了捋平时工作会遇到的一些常见问题,首先想到了多用户登录限制问题,下面就对此问题做一点思考讲解。
- ZXing是谷歌的一个开源库,可以用来生成二维码、扫描二维码。本文所介绍的是第一部分。首先上效果图:ZXing相关各种文件官方下载地址:ht
- 本文实例为大家分享了Spring Boot实现文件上传下载的具体代码,供大家参考,具体内容如下示例【Spring Boot 文件
- 判断用户输入的是否至少含有N位小数。1.当用户输入的是非数字时抛出异常,返回false。2.当用户输入数字是,判断其数字是否至少含有N位小数
- 单元测试是程序员对代码的自测,一般公司都会严格要求单元测试,这是对自己代码的负责,也是对代码的敬畏。一般单元测试都是测试Service层,下
- 作者:精致码农出处:http://cnblogs.com/willick联系:liam.wang@live.com最近工作中遇到一个这样的需
- 本文实例讲述了Java swing框架实现的贪吃蛇游戏。分享给大家供大家参考,具体如下:java是门高级语言,做游戏时适合做后台,但是用它也
- 一直想在持续集成方向学习并研究一番,近期正准备结合jmeter+ant+jenkins做自动化接口测试,在学习的同时,正好实践一番,毕竟实践
- 简述在学着使用Java的命令行来编译java文件的时候,遇到了这个问题Windows操作系统报错“‘
- 一、前言目前大部分手机都是 60Hz 的刷新率,也就是 16.6ms 刷新一次,系统为了配合屏幕的刷新频率,将 Vsync 的周期也设置为
- 前言我们都知道,kill在linux系统中是用于杀死进程。kill pid [..]kill命令可将指定的信号发送给相应的进程或工作。 ki