【C++11】volatile 关键字

文章目录

  • 一、基本作用

  • 二、应用场景

    • [2.1 硬件寄存器访问](#2.1 硬件寄存器访问)
    • [2.2 多线程环境中的信号处理](#2.2 多线程环境中的信号处理)
    • [2.3 内存映射 I/O](#2.3 内存映射 I/O)
  • [三、不用 volatile 导致的问题](#三、不用 volatile 导致的问题)

    • [3.1 死循环问题](#3.1 死循环问题)
    • [3.2 编译器优化导致的逻辑错误](#3.2 编译器优化导致的逻辑错误)
  • 四、小结

  • C++11 之前,volatile 常被误用于多线程同步;C++11 引入标准内存模型和 std::atomic 后,volatile 被明确限定仅用于硬件寄存器、信号处理等非并发场景,不再适用于多线程同步。

一、基本作用

  • volatile 是 C++ 中的一个类型限定符(type qualifier),用于告诉编译器: 该变量的值可能会在程序的控制之外被改变 ,因此 编译器不能对该变量进行某些优化(防止编译器将变量缓存在寄存器中),每次使用变量时都必须从内存中重新读取。 确保对变量的修改对其他可能修改它的实体可见。

二、应用场景

2.1 硬件寄存器访问

cpp 复制代码
// 访问硬件状态寄存器
volatile uint32_t* status_register = reinterpret_cast<volatile uint32_t*>(0x40000000);

while ((*status_register & 0x1) == 0) {
    // 等待硬件就绪标志位被置位
    // ⚠️如果没有 volatile,编译器可能只读取一次并缓存结果
}

2.2 多线程环境中的信号处理

cpp 复制代码
volatile bool signal_received = false;

void signal_handler(int sig) {
    signal_received = true;  // 信号处理器修改变量
}

int main() {
    signal(SIGINT, signal_handler);
    while (!signal_received) {
        // 执行工作
        // ⚠️必须使用 volatile,否则编译器可能优化掉重复检查
    }
    return 0;
}

2.3 内存映射 I/O

cpp 复制代码
// 内存映射的显示缓冲区
volatile char* display_buffer = reinterpret_cast<volatile char*>(0xA0000000);
for (int i = 0; i < 100; ++i) {
    display_buffer[i] = 'A' + i;  // 每次写入都必须实际执行
}

三、不用 volatile 导致的问题

3.1 死循环问题

cpp 复制代码
// ❌️错误示例:没有使用 volatile
bool flag = false;

void interrupt_handler() {
    flag = true;  // 在中断或信号处理器中设置
}

void wait_for_flag() {
    while (!flag) {
        // ⚠️ 编译器可能优化为:只读取一次 flag 的值
        // 如果初始值为 false,就会陷入死循环!
    }
}

// ✅️正确做法:
volatile bool flag = false;  // ✅️ 添加 volatile

3.2 编译器优化导致的逻辑错误

cpp 复制代码
// 模拟硬件寄存器

// ❌️错误:没有 volatile
uint32_t hardware_status = 0;
void wait_for_hardware() {
    while (hardware_status == 0) {
        // ⚠️编译器可能将 hardware_status 缓存在寄存器中
        // 即使硬件改变了内存中的值,循环也不会退出
    }
}

// ✅️正确:使用 volatile
volatile uint32_t hardware_status = 0;
void wait_for_hardware_correct() {
    while (hardware_status == 0) {
        // 每次都会从内存重新读取 hardware_status
    }
}

四、小结

  • volatile ≠ 线程安全,volatile 不能保证原子性。
cpp 复制代码
// volatile 不能保证原子性!
volatile int shared_var = 0;

// 在多线程中,shared_var++ 仍然不是原子操作
// 需要使用 std::atomic 或互斥锁
  • 性能影响
    • 每次访问都必须访问内存,无法使用寄存器缓存
    • 可能影响性能,只在必要时使用
相关推荐
BirdenT18 小时前
20260519紫题训练
c++·算法
Highcharts.js1 天前
倒置百分比堆叠面积图表示列详解|Highcharts大气成分图表代码
开发语言·信息可视化·highcharts·图表开发·面积图·图表示例·推叠图
csdn_aspnet1 天前
C语言 Lomuto分区算法(Lomuto Partition Algorithm)
c语言·开发语言·算法
晨曦中的暮雨1 天前
4.15腾讯 CSIG云服务产线 一面
java·开发语言
存在morning1 天前
【GO语言开发实践】二 GO 并发快速上手
大数据·开发语言·golang
xiaoerbuyu12331 天前
开源Java 邮箱 基于SpringBoot+Vue前后端分离的电子邮件
java·开发语言
C+++Python1 天前
C++ 进阶学习完整指南
java·c++·学习
sparEE1 天前
c++值类别、右值引用和移动语义
开发语言·c++
zhangjw341 天前
第11篇:Java Map集合详解,HashMap底层原理、哈希冲突、JDK1.8优化、遍历方式彻底吃透
java·开发语言·哈希算法
jrrz08281 天前
Apollo MPC Controller
c++·自动驾驶·apollo·mpc·横向控制·lateral control