C/C++的volatile关键字应用示例

首先必须强调volatile无法用来保证线程安全。

volatile的功能是阻止编译器优化,从而直接从内存中读写变量的值。由于操作系统访问寄存器的速度远大于访问内存的速度(两者之间还有各级cache),所以编译程序时可能会进行优化,比如这样的语句

int flag = 1;
while (flag) {}

由于多次对flag进行判断,所以编译器可能优化为把flag变量的值从内存中拷贝到寄存器中,之后每次都从寄存器中读取。从代码的层面看起来没有问题,但是比如信号处理函数改变了flag的值,更新的flag不会立刻(甚至不会)反映到寄存器中,因此读取的flag的值还是旧的值。
给出示例代码

// test.cc
#include <stdio.h>
#include <signal.h>

static int g_iRun = 1;

void sigint_handler(int) { g_iRun = 0; }

int main() {
    if (signal(SIGINT, sigint_handler) == SIG_ERR)
        perror("signal SIGINT");
    while (g_iRun) {}
    printf("sigint caught!\n");
    return 0;
}
$ g++ test.cc 
$ ./a.out 
^Csigint caught!
$ g++ test.cc -O
$ ./a.out 
^C^C^C^C^\Quit (core dumped)

可以看到仅仅加了-O选项,最低层次的优化下信号处理器都不会对Ctrl+C做出反应。
为了防止程序直接从寄存器中读取变量的值,需要用volatile来修饰g_iRun变量
static volatile int g_iRun = 1;
修饰之后,即使用-O3选项进行优化,仍然可以捕捉信号

$ g++ test.cc -O3
$ ./a.out 
^Csigint caught!

另一个典型应用就是APUE上图7-13的示例程序,C程序使用setjmplongjmp回滚函数栈帧时,自动变量的值是否回滚是不确定的。

// test.cc
#include <stdio.h>
#include <setjmp.h>

static jmp_buf jmpbuffer;

void func() { longjmp(jmpbuffer, 1); }

int main() {
    int x = 1;
    if (setjmp(jmpbuffer) == 0) {  
        x = 2;
        func();
    } else {  // 从longjmp中返回
        printf("%d\n", x);
    }
    return 0;
}
$ g++ test.cc 
$ ./a.out 
2
$ g++ test.cc -O
$ ./a.out 
1

和处理信号的示例一样,用了优化选项-O编译后,自动变量x的值在longjmp后从2变成了1。如果用volatile修饰自动变量x,那么longjmp之后x的值保证为2。

最后说说为什么无法保证线程安全。
线程安全指在多线程环境下,无论多线程如何交替执行,最后的结果都是预期值。比如N个线程对变量x执行x = x + 1操作,最后x的值增加了N。
举个经典例子,2个线程对volatile变量x(初值为0)执行自增操作,既可能是这样的执行顺序

  1. 线程A从内存中读取x的值(A.x=0);
  2. 线程B从内存中读取x的值(B.x=0);
  3. 线程A写入值(A.x+1=1)到x的内存中(x此时为1);
  4. 线程B写入值(B.x+1=1)到x的内存中(x此时为1);

也有可能时这样的执行顺序

  1. 线程A从内存中读取x的值(A.x=0);
  2. 线程A写入值(A.x+1=1)到x的内存中(x此时为1);
  3. 线程B从内存中读取x的值(B.x=1);
  4. 线程B写入值(B.x+1=2)到x的内存中(x此时为2);

两种不同的执行顺序导致了不同的结果,但是两个线程都是直接从内存中读写变量x。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351