C++11 中的双重检查锁单例

2024-03-03

以下单例实现是否没有数据竞争?

static std::atomic<Tp *> m_instance;
...

static Tp &
instance()
{
    if (!m_instance.load(std::memory_order_relaxed))
    {
        std::lock_guard<std::mutex> lock(m_mutex);
        if (!m_instance.load(std::memory_order_acquire))
        {
            Tp * i = new Tp;
            m_instance.store(i, std::memory_order_release);    
        }    
    }

    return * m_instance.load(std::memory_order_relaxed);
}

Is the std::memory_model_acquire负载操作是否多余?是否可以通过将加载和存储操作切换为来进一步放松它们std::memory_order_relaxed?在这种情况下,获取/释放语义是std::mutex足以保证其正确性,或者进一步std::atomic_thread_fence(std::memory_order_release)还需要确保构造函数的内存写入发生在宽松存储之前?然而,使用围栏是否等同于让商店拥有memory_order_release?

EDIT:感谢约翰的回答,我想出了以下应该没有数据竞争的实现。尽管内部负载可能根本不是原子的,但我决定保留宽松的负载,因为它不会影响性能。与始终具有获取内存顺序的外部负载相比,thread_local 机制将访问实例的性能提高了大约一个数量级。

static Tp &
instance()
{
    static thread_local Tp *instance;

    if (!instance && 
        !(instance = m_instance.load(std::memory_order_acquire)))
    {
        std::lock_guard<std::mutex> lock(m_mutex);
        if (!(instance = m_instance.load(std::memory_order_relaxed)))
        {
            instance = new Tp; 
            m_instance.store(instance, std::memory_order_release);    
        }    
    }
    return *instance;
}

我认为这是一个很好的问题,约翰·卡尔斯贝克有正确的答案。

然而,需要明确的是,惰性单例最好使用经典的 Meyers 单例来实现。它保证了 C++11 中的正确语义。

§ 6.7.4

...如果控制进入 变量初始化时同时声明,并发执行需等待 完成初始化。 ...

Meyer 的单例是首选,因为编译器可以积极优化并发代码。如果编译器必须保留 a 的语义,它会受到更多限制std::mutex。此外,迈耶的单例是2 lines而且几乎不可能出错。

这是迈耶单例的经典示例。简单,优雅,又在c++03中破碎。但 c++11 简单、优雅、功能强大。

class Foo
{
public:
   static Foo& instance( void )
   {
      static Foo s_instance;
      return s_instance;
   }
};
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++11 中的双重检查锁单例 的相关文章

随机推荐

  • 自定义 STL 序列的最小嵌套 typedef 集?

    应在满足序列概念的自定义 STL 类中定义的最小嵌套 typedef 集是什么 自定义序列应兼容 std back insert iterator http www sgi com tech stl front insert iterato
  • 从 Apple Developer Portal 中删除开发者证书

    我错误地在我的 Apple 开发者帐户上创建了开发者证书 现在我想从门户中删除它 但我不喜欢任何方法来删除它 您能给我一个如何从苹果开发者门户删除证书的想法 建议吗 如果您错误地创建了证书或者想要重置证书 您可以从 Apple Develo
  • 如何连接位置数据(纬度、经度)

    我必须数据集 一个包含某个位置 经纬度 即测试 另一个包含纽约市所有邮政编码的纬度 经度信息 即 test2 test lt structure list trip count 1 10 dropoff longitude c 73 959
  • AWS Lambda 无法删除 Amazon S3 对象

    我正在尝试创建一个 AWS Lambda 函数 该函数处理上传到第一个存储桶的文件 然后将其保存到第二个存储桶 然后删除输入文件 问题是 当我尝试删除文件时 我得到了 message Access Denied code AccessDen
  • 复制的 std::list 中的垃圾

    我的图形类看起来像 class Graph public typedef unsigned int size type typedef std list
  • 具有静态存储持续时间的对象构造函数中的 std::cout 使用

    使用安全吗std cout在 C 98 C 03 中具有静态存储持续时间的对象的构造函数中 由此看来answer https stackoverflow com a 8785008 1608835它不是 但它不包含标准中的任何引用 只有在
  • 在另一个 for 循环中使用 for 循环

    我正在尝试以彩虹色打印文件 但是我有一个问题 这是我的代码 color 91 93 92 96 94 95 with open sys argv 1 as f for i in f read for c in color print 033
  • 哈希图与数组性能

    当数组的索引已知时 使用数组或 HashMap 是否 性能方面 更好 请记住 示例中的 对象数组 映射 只是一个示例 在我的实际项目中 它是由另一个类生成的 因此我不能使用单独的变量 数组示例 SomeObject objects new
  • PHP 和 mySQL 单引号还是双引号?

    我没有看到使用单引号或双引号有任何区别 我何时或为什么应该使用第一个而不是后者 我读过一些关于 SQL 我应该使用单打的文章 为什么 您应该使用单引号和mysql real escape string http php net manual
  • 在表格行上使用 Flexbox?

    我找不到任何快速答案 使用起来爽吗display flex 在表行上 tr 元素 感觉不对 但如果没有兼容性问题我会这样做 这是我正在谈论的内容的代码笔 https codepen io connorv pen boEYOB editors
  • 如何在不擦除 \perl\site\lib\ 和 \perl\site\bin\ 的情况下升级 Strawberry Perl?

    我下载了 5 12 3 安装程序来升级 5 12 1 安装 安装5 12 3后 我之前安装的cpan模块消失了 这是预期的行为吗 我应该在升级之前备份我的库以防止这种情况发生吗 我会声称这不应该是预期的行为 The site目录不是标准发行
  • 如何使 JsonResult 返回数组数组(不带字段名)而不是对象数组?

    我有一个日期 值对的 IEnumerable 列表 我将其作为 Json 列表返回以进行浮动 但是 当我调用 JsonResult 时 结果如下所示 Date date1 Value value1 Date date2 Value valu
  • Angular 1.2 的 debugInfoEnabled

    Angular 1 3引入了一个新的debugInfoEnabled https docs angularjs org api ng provider 24compileProvider debugInfoEnabled调用该方法可以提高性
  • 这是使用 jQuery 解析 XML 时的错误吗?

    好的 我会尝试尽可能简单地解释这一点 我正在尝试使用 jQuery 解析通过 Web 服务获得的 XML 一切都按其应有的方式进行 直到我注意到看在上帝的份上 我无法解析名为 image 的节点 长话短说 2小时后 我注意到 问题出在标签名
  • 如何解决文档未定义错误?

    我收到此错误 我以为是因为 node js 但我不确定 我该如何解决它 Running node c Users Lenovo Desktop projectjs index js c Users Lenovo Desktop projec
  • 理解 Haskell as 模式

    我正在通读 Real World Haskell 并试图理解 as 模式 摘自书中 第 4 章 suffixes a gt a suffixes xs xs xs suffixes xs suffixes 这本书解释了 符号 将变量 xs
  • 如何替换特定文本行中的特定单词[关闭]

    Closed 这个问题需要多问focused help closed questions 目前不接受答案 好的 我有这个 yaml 文件 我想替换一个字符串 socialspy true 用字符串 socialspy false 但我不知道
  • 如何以编程方式判断安装了哪些 Outlook 插件以及它们是否已启用?

    如何确定安装了哪些 Outlook COM 或 PIA 插件以及它们是否已启用 我怎样才能获得这些信息 以及文件版本 1 如果您想从另一个 Outlook 加载项内部访问此信息 您可以使用 Application ComAddins 对象
  • LDAP 过滤器 - 查找特定 OU 的所有用户

    我遇到了问题LDAP Search Filter 我需要检索的是特定的所有用户LDAP组即OU Staff OU Users OU Accounts DC test DC local 我的搜索是 objectCategory user OU
  • C++11 中的双重检查锁单例

    以下单例实现是否没有数据竞争 static std atomic