IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> Item 16: Make const member functions thread safe. -> 正文阅读

[C++知识库]Item 16: Make const member functions thread safe.

Effective Modern C++ Item 16 的学习和解读。

const 成员的好处

一般地,对于不修改成员变量的函数申明为 const 类型,它隐含了是被传入的 this 指针为 const 类型。const 成员函数有两个好处:

  • 不会修改成员变量(有特例,下面会介绍),比较安全。
  • 可以被 const 和 非 const 对象调用。非 const 成员变量无法被 const 对象调用。

由于 const 成员函数一般不会修改成员变量,因此 const 成员函数一般也是线程安全的。

const 成员函数的线程安全问题

凡是皆有例外。

考虑这样一个场景:用一个类来表示多项式,并求解其根。求根的过程不会修改多项式本身,因此我们可以设计如下:

class Polynomial {
public:
  using RootsType = std::vector<double>;
  …
  RootsType roots() const; // 此处省略其实现};

从功能角度来说,上面的设计没有问题。但是,求解根的代价一般都比较高,我们一般不希望重复计算多次。因此,我们可以将多项式的根缓存起来,只在必要的时候才计算,可以修改如下:

class Polynomial {
public:
  using RootsType = std::vector<double>;
  RootsType roots() const
  {
    if (!rootsAreValid) { // if cache not valid compute roots, store them in rootVals
      … 
      rootsAreValid = true;
    }
    return rootVals;
  }
private:
  mutable bool rootsAreValid{ false };
  mutable RootsType rootVals{};
};

从概念上说,求解根不会修改多项式本身。但是,缓存的行为又会修改多项式类的成员变量。经典的做法就是将成员变量申明为 mutable,成员函数申明为 const。

如果 const 成员函数不会被多个线程调用,那上述代码没有任何问题。

但是,当存在两以上的对象同时调用 const 成员变量,则会发生线程安全问题:这两个线程中的一个或两个都可能尝试去修改成员变量 rootsAreValid 和 rootVals。这意味着在没有同步的情况下,两个不同的线程读写同一段内存,这段代码会会有未定义的行为。

const 成员函数线程安全保证

很容易想到的解决办法是加 mutex:

class Polynomial {
public:
  using RootsType = std::vector<double>;
  RootsType roots() const
  {
    std::lock_guard<std::mutex> g(m); // lock mutex
    if (!rootsAreValid) { // if cache not valid, compute/store roots
      … 
      rootsAreValid = true;
    }
    return rootVals;
  } // unlock mutex
private:
  mutable std::mutex m;
  mutable bool rootsAreValid{ false };
  mutable RootsType rootVals{};
};

值得注意的是,由于 std::mutex 是 move-only 类型,所以包含了 std::mutex 变量的 Polynomial 对象也是 move-only 类型,失去了 copy 的能力。

在有些情况下,使用 std::mutex 的代价可能是比昂贵的。如果你只需要同步单一变量或者内存单元,使用 std::atomic 是相对比较廉价的。举个例子,你只需要统计一个成员函数被调用的次数:

class Point { // 2D point
public:double distanceFromOrigin() const noexcept // see Item 14 for noexcept
  {
    ++callCount; // atomic increment
    return std::sqrt((x * x) + (y * y));
  }
private:
  mutable std::atomic<unsigned> callCount{ 0 };
  double x, y;
};

和 std::mutex 一样, std::atomic 也是 move-only 的。

虽然 std::atomic 的开销一般比 std::mutex 高,但对于多于一个变量或内存块需要同步,还是应该使用 std::mutex。因此,看你的需求,选择使用 std::mutex 或者 std::atomic 。

总结一下:

  • 让 const 成员函数做到线程安全,除非你确保它们永远不会用在多线程的环境下。
  • 相比于 std::mutex,使用 std::atomic 变量能提供更好的性能,但是它只适合处理单一的变量或内存单元。
  • std::mutex 和 std::atomic 都是 move-only 的。
  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-03-08 22:10:02  更:2022-03-08 22:11:32 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 10:15:49-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码