当前位置: 首页 > news >正文

C++11原子操作:从入门到精通

文章目录

    • 一、什么是原子操作?
    • 二、为什么需要原子操作?
    • 三、C++11中的<atomic>头文件
    • 四、基本使用
      • 1. 声明原子变量
      • 2. 基本原子操作
    • 五、内存顺序(Memory Order)
      • 示例:使用内存顺序实现自旋锁
    • 六、原子类型模板
    • 七、实际应用示例
      • 1. 线程安全的计数器
      • 2. 双重检查锁定(Double-Checked Locking)
    • 八、性能考虑
    • 九、常见陷阱
    • 十、总结

一、什么是原子操作?

原子操作(Atomic Operations)是指不可被中断的一个或一系列操作。在多线程编程中,原子操作就像是"不可分割的最小单位",要么完全执行,要么完全不执行,不会出现执行到一半被其他线程打断的情况。

二、为什么需要原子操作?

考虑以下场景:

int counter = 0;// 线程1
void increment() {counter++;
}// 线程2
void decrement() {counter--;
}

在多线程环境下,counter++counter-- 这样的操作实际上包含多个步骤(读取、修改、写入),可能导致竞态条件(Race Condition)。原子操作可以确保这些操作不可分割地完成。

三、C++11中的头文件

C++11引入了<atomic>头文件,提供了一系列原子类型和操作。主要包含:

  1. 基本原子类型(atomic_int, atomic_bool等)
  2. 类模板std::atomic<T>
  3. 原子操作函数
  4. 内存顺序控制

四、基本使用

1. 声明原子变量

#include <atomic>
#include <iostream>int main() {std::atomic<int> counter(0);  // 初始化为0counter.store(10);  // 原子存储int value = counter.load();  // 原子读取std::cout << "Counter: " << value << std::endl;return 0;
}

2. 基本原子操作

std::atomic<int> num(0);// 原子加法
num.fetch_add(5);  // 相当于 num += 5// 原子减法
num.fetch_sub(3);  // 相当于 num -= 3// 原子交换
int old = num.exchange(10);  // 将num设为10,返回旧值// 比较交换(CAS操作)
int expected = 10;
bool success = num.compare_exchange_strong(expected, 15);
// 如果num == expected,则设为15,返回true
// 否则将expected设为num的实际值,返回false

五、内存顺序(Memory Order)

内存顺序指定了原子操作周围的内存访问如何排序,是原子操作的高级特性。C++11定义了6种内存顺序:

  1. memory_order_relaxed:最宽松的顺序,只保证原子性
  2. memory_order_consume:依赖于此原子变量的后续操作不能重排序到它前面
  3. memory_order_acquire:后续操作不能重排序到它前面
  4. memory_order_release:前面操作不能重排序到它后面
  5. memory_order_acq_rel:acquire + release
  6. memory_order_seq_cst:最严格的顺序(默认)

示例:使用内存顺序实现自旋锁

#include <atomic>
#include <thread>class SpinLock {std::atomic_flag flag = ATOMIC_FLAG_INIT;public:void lock() {while (flag.test_and_set(std::memory_order_acquire)) {// 自旋等待}}void unlock() {flag.clear(std::memory_order_release);}
};

六、原子类型模板

std::atomic可以用于任何可平凡复制的类型:

struct Point {int x, y;
};std::atomic<Point> atomic_point;void updatePoint() {Point new_point{10, 20};atomic_point.store(new_point, std::memory_order_release);
}void readPoint() {Point current = atomic_point.load(std::memory_order_acquire);std::cout << "Point: (" << current.x << ", " << current.y << ")\n";
}

七、实际应用示例

1. 线程安全的计数器

#include <atomic>
#include <thread>
#include <vector>
#include <iostream>std::atomic<int> counter(0);
const int kIncrementsPerThread = 100000;
const int kThreadCount = 10;void increment() {for (int i = 0; i < kIncrementsPerThread; ++i) {counter.fetch_add(1, std::memory_order_relaxed);}
}int main() {std::vector<std::thread> threads;for (int i = 0; i < kThreadCount; ++i) {threads.emplace_back(increment);}for (auto& t : threads) {t.join();}std::cout << "Final counter value: " << counter << std::endl;return 0;
}

2. 双重检查锁定(Double-Checked Locking)

class Singleton {
private:static std::atomic<Singleton*> instance;static std::mutex mutex;Singleton() {}public:static Singleton* getInstance() {Singleton* tmp = instance.load(std::memory_order_acquire);if (tmp == nullptr) {std::lock_guard<std::mutex> lock(mutex);tmp = instance.load(std::memory_order_relaxed);if (tmp == nullptr) {tmp = new Singleton();instance.store(tmp, std::memory_order_release);}}return tmp;}
};std::atomic<Singleton*> Singleton::instance(nullptr);
std::mutex Singleton::mutex;

八、性能考虑

  1. 原子操作比普通操作慢,但比互斥锁快
  2. 使用memory_order_relaxed可以获得更好性能(当不需要严格顺序时)
  3. 避免过度使用原子变量,考虑无锁数据结构设计

九、常见陷阱

  1. 错误地认为原子操作可以替代锁:原子操作只保证单个操作的原子性,复杂操作仍需锁
  2. 忽略内存顺序:错误的内存顺序可能导致难以发现的bug
  3. ABA问题:在使用CAS操作时需要注意

十、总结

C++11的原子操作提供了:

  1. 线程安全的基本操作
  2. 多种内存顺序控制
  3. 比锁更高效的并发控制方式
  4. 构建无锁数据结构的基础

掌握原子操作是成为高级C++开发者的重要一步。建议从简单场景开始实践,逐步理解更复杂的内存顺序概念。

相关文章:

  • 独立站 wordpress天津百度seo
  • 西安mg动画制作网站建设互联网营销专业
  • ocr是不是用于制作网页的软件快速提升排名seo
  • 大神自己做的下载音乐的网站产品推广思路
  • 企业做网站和开展电子商务的好处百度流量
  • 如何做网站活动青岛招聘seo
  • 西游记12:观世音菩萨送袈裟和禅杖;菩萨现身,教导大乘佛法三藏;御弟圣僧;宁恋本乡一捻(niǎn)土,莫爱他乡万两金。
  • 基于Hp感染的慢性胃炎居家管理小程序的设计与实现(消息震动)
  • DuDuTalk | 武汉赛思云科技有限公司通过武汉市人工智能企业认定!
  • 掌握CIS基准合规性:通过自动化简化网络安全
  • sentinel 自定义 dashboard 用户名密码
  • 【网站内容安全检测】之1:获取网站所有链接sitemap数据
  • 5.1 基于livox_ros_driver2运行MID360demo
  • 基于LangChat搭建RAG与Function Call结合的聊天机器人方案
  • 卷积神经网络(Convolutional Neural Network, CNN)
  • 1688商品发布API:自动化上架与信息同步
  • 多传感器标定简介
  • 快速排序算法
  • 设计模式精讲 Day 13:责任链模式(Chain of Responsibility Pattern)
  • 【沉浸式解决问题】微服务子模块引入公共模块的依赖后无法bean未注入
  • 【笔记】Docker 配置阿里云镜像加速(公共地址即开即用,无需手动创建实例)
  • 14.Linux Docker
  • Web层注解
  • dovi交叉编译方法(编译libdovi.so)
  • selenium4中Chrome浏览器显示浏览器受自动化测试软件控制的解决方法
  • 打造丝滑的Android应用:LiveData完全教程