当前位置:首页 > 技术文章 > 正文内容

C/C++面试题(二):std::atomic与volatile

volatile 是 C/C++ 中的一个关键字,用于告知编译器某个变量的值可能会在程序的控制之外被意外修改(例如被硬件、中断服务程序、多线程环境或其他外部代理)。为了防止编译器对代码进行某些可能破坏程序正确性的优化,开发者需要使用 volatile 来强制编译器在每次访问该变量时都直接从内存中读取或写入,而不是依赖寄存器中的缓存值或优化掉“看似冗余”的访问。

std::atomic 是 C++11 引入的一个模板类,用于在多线程环境中实现原子操作。它确保对共享变量的操作是线程安全的,并且可以控制内存顺序(memory order)。

volatile

  1. 禁用编译器优化
  • 编译器通常会对变量访问进行优化,例如将变量缓存到寄存器中以提高访问速度。
  • volatile 会强制编译器每次访问变量时都从内存中读取或写入,确保获取最新值。
  1. 典型应用场景:
  • 硬件寄存器:嵌入式系统中,硬件状态可能通过内存映射的寄存器被外部修改。
// 假设 0x4000 是某个硬件寄存器的内存映射地址
volatile uint32_t* const hardware_reg = reinterpret_cast<volatile uint32_t*>(0x4000);

void read_hardware() {
    // 每次读取都会直接从硬件寄存器获取最新值
    uint32_t value = *hardware_reg;
}
  • 中断服务程序(ISR):中断可能修改共享变量。
volatile bool data_ready = false; // 中断可能修改此变量

void ISR() {
    data_ready = true; // 中断触发时修改
}

int main() {
    while (!data_ready) { // 每次循环都从内存读取最新值
        // 等待数据准备就绪
    }
}
  • 多线程共享变量(但需注意,volatile 不能替代线程同步机制,多线程中仍需使用 std::atomic 或互斥锁)。
volatile bool flag = false;

// 线程 A
void thread_a() {
    flag = true; // 修改标志
}

// 线程 B
void thread_b() {
    while (!flag) { // 读取标志
        // 等待
    }
}

std::atomic的常用操作

  • 加载(Load)
int value = counter.load(std::memory_order_relaxed);
  • 存储(Store)
counter.store(42, std::memory_order_relaxed);
  • 交换(Exchange)
int old_value = counter.exchange(100);
  • 比较并交换(Compare-and-Swap, CAS)
int expected = 10;
bool success = counter.compare_exchange_strong(expected, 20);
  • 原子加减
counter.fetch_add(5);  // 原子加 5
counter.fetch_sub(3);  // 原子减 3

内存顺序(Memory Order)

std::atomic 支持多种内存顺序,用于控制操作的可见性和顺序性。常用的内存顺序包括:

  • std::memory_order_relaxed:只保证原子性,不保证顺序。
  • std::memory_order_acquire:保证后续操作不会重排到该操作之前。
  • std::memory_order_release:保证前面的操作不会重排到该操作之后。
  • std::memory_order_seq_cst:最严格的内存顺序,保证全局顺序一致性。
counter.store(42, std::memory_order_release);
int value = counter.load(std::memory_order_acquire);

std::atomic应用场景

  • std::atomic 是 C++ 中实现线程安全操作的首选工具
  • 它提供了原子操作和内存顺序控制,适用于多线程环境中的共享变量。
  • 与 volatile 不同,std::atomic 不仅保证原子性,还提供了更强的内存顺序保证。

std::atomic与volatile的区别总结

特性

volatile

std::atomic

禁用编译器优化

原子性保证

内存顺序控制

支持(如 memory_order_relaxed)

适用场景

硬件/中断/信号处理

多线程同步

扫描二维码推送至手机访问。

版权声明:本文由AR编程网发布,如需转载请注明出处。

本文链接:http://www.arlanguage.com/post/4260.html

分享给朋友:

“C/C++面试题(二):std::atomic与volatile” 的相关文章

nginx 多域名配置 nginx多站点配置示例

Nginx 可以配置多个域名,以便根据不同的域名来处理不同的请求。下面是一个配置多个域名的例子:server { listen 80; server_name example1.com; location / { root /var/www/example1...

http请求域名强制跳转https

网站添加了https证书后,当http方式访问网站时就会报404错误,所以需要做http到https的强制跳转设置. ---------------一、采用nginx的rewrite方法--------------------- 下面是将所有的http请求通过rewrite重写到https上。 例如...

压测nginx出现的问题分析

压测nginx出现no live upstreams while connecting to upstream的问题分析基础环境版本信息Centos 7.1nginx version: openresty/1.13.6.2nginx配置信息stream {   ...

http——使用Nginx部署HTTPS服务

https 服务部署使用 HTTPS 需要生成私钥与公钥;某个文件夹中打开 git bash 输入命令:openssl req -x509 -newkey rsa:2048 -nodes -sha256 -keyout localhost-privkey.pem -out localhost-cer...

nginx入门——web服务器(四)

我们先来认识一下web服务器主要功能。处理哪些URL请求怎么处理这些URL请求下面我们主要针对这两个功能来进行讲解。一、设置虚拟服务器在http上下文中用server指令定义虚拟服务器,以下是一个基本的结构:http { server { # Server上下文配置...

linux和nginx,https配置实战精辟总结

文章目录前言1.linux服务器常用安装:1.1通过yum安装rz和sz命令:1.2 yum安装docker:2. nginx安装:2.1 nginx配置文件:2.2 服务配置https文件:2.3 总结3.结尾:前言现在我们如果自己想弄点东西,需要买自己的云服务器、域名,整一个https的域名进行...