单例模式之模板与跨库(DLL、SO)


前言

单例模式用模板实现的话比较优雅方便,基本不需要写额外的代码。但也有个问题,就是封装成库之后,比如windows下是封装成DLL,那么DLL内的单例和外部单例将会是2个不同的实例对象,可以做实验发现2个指针是不一样的地址。这篇文章就是总结一下用模板实现单例,并且可以跨库是怎么做到的。


一、单例模板范式

代码实例也都是从网上抄来的,没什么好解释的,就一点要说明下,模板的实现全部放头文件(包括单例中的静态成员变量),不然会报错:

template <typename T>
class Singleton
{
public:
	static std::shared_ptr<T> instance()
	{
		static std::once_flag flag;
		std::call_once(flag, []() {
			m_instance.reset(new T);
			});
		return m_instance;
	}
private:
	Singleton() = default;
	Singleton(const Singleton&) = delete;
	Singleton& operator=(const Singleton&) = delete;
	static std::shared_ptr<T> m_instance;
};
template<typename T>
std::shared_ptr<T> Singleton<T>::m_instance;

二、使用宏定义声明单例类

1.宏定义

这个宏的作用是禁止外部使用构造函数声明对象,并且添加instance()接口。大家可能会觉得有点像脱裤子放屁,但这都是为了跨库使用,并且尽可能的优美,看到最后就明白了。如果不考虑跨库使用,上面的单例模板已经做到了,

#define DECL_SINGLETON(T)\
private:\
explicit T() {}\
friend class Singleton<T>;\
public:\
	static std::shared_ptr<T> instance()\
	{\
		return Singleton<T>::instance();\
	}

2.声明单例类

代码如下(示例):

class ClassificationInterface
{
	DECL_SINGLETON(ClassificationInterface)
public:
	void func(); //do some thing
};

这样看起来是不是很简洁,使用的时候这个类就是单例了直接可以这样:

ClassificationInterface::instance()->func();

增加导出符号

最后,我们将这个类导出,就可以跨库使用了,原理是:我们宏定义的instance()函数会被导出,这个函数本身不是模板,而是将模板实例化了,是有唯一性的。关于导出符号不懂的搜索下吧。

#ifndef TEST_WITHOUT_WRAP_API
#if WIN32
#ifdef EXPORT_WRAP_API
#define WRAP_API __declspec(dllexport)
#else
#define WRAP_API __declspec(dllimport)
#endif
#else
#define WRAP_API __attribute__ ((visibility ("default")))
#endif
#else
#define WRAP_API
#endif

class WRAP_API ClassificationInterface
{
	DECL_SINGLETON(ClassificationInterface)
public:
	void func(); //do some thing
};

总结

总结了单例模式的模板跨库范式,最后在类中增加一个宏定义即可完成单例类声明。个人感觉实现较为优雅。有兴趣的可以做个实验,看看单例指针是不是唯一的。

  • 1
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
单例模式是一种设计模式,用于确保一个类只有一个实例,并提供全局访问点。在C++中,可以通过以下步骤实现单例模式: 1. 创建一个类,将其构造函数、拷贝构造函数和赋值运算符声明为私有,以防止外部直接创建或复制实例。 2. 在类中定义一个静态成员变量,用于保存类的唯一实例。 3. 提供一个公共的静态成员函数,用于获取该唯一实例。在此函数中,检查静态成员变量是否已经被初始化,如果没有,则创建一个新实例并返回。 4. 使用单例类时,只需调用公共的静态成员函数即可获得类的唯一实例。 关于DLL(动态链接库),在使用单例模式时需要注意以下几点: 1. 单例类的实例在DLL中是唯一的,即使在同一个进程中加载了多个相同的DLL,也只有一个实例。 2. 需要确保DLL中的单例实例在整个进程中是可见且可访问的。可以将单例实例声明为导出函数,并使用导出关键字(如`__declspec(dllexport)`)导出它。 3. 在使用单例实例时,需要通过导入关键字(如`__declspec(dllimport)`)导入DLL中的单例实例。 需要注意的是,单例模式并不是线程安全的。在多线程环境下,可能会导致多个线程同时创建实例,破坏单例的初衷。因此,在实现单例模式时,需要考虑线程安全性,可以使用互斥锁或双重检查锁定等机制来保证线程安全。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值