C 参考手册
- C 语言
- C 的历史
- 基本概念
- 表达式
- 声明
- 初始化
- 函数
- 语句
- 静态断言
- 字符常量
- 函数声明
- 函数定义
- 转义序列
- 翻译阶段
- 标识符
- 作用域
- 生存期
- 查找与命名空间
- ASCII 码表
- 类型
- 遵从性
- 算术类型
- restrict 类型限定符
- 类型
- 对象与对齐
- 主函数
- 未定义行为
- 内存模型
- if 语句
- switch 语句
- for 循环
- while 循环
- do-while 循环
- continue语句
- break 语句
- goto语句
- return 语句
- 值类别
- 求值顺序
- 整数常量
- 浮点常量
- 字符串字面量
- 复合字面量
- 常量表达式
- 隐式转换
- 成员访问运算符
- 逻辑运算符
- 比较运算符
- 算术运算符
- 赋值运算符
- 自增/自减运算符
- 其他运算符
- sizeof 运算符
- _Alignof 运算符
- 转型运算符
- C 运算符优先级
- 泛型选择
- 标量初始化
- 数组初始化
- 结构体与联合体初始化
- 指针声明
- 数组声明
- 枚举
- 存储类指定符
- const 类型限定符
- volatile 类型限定符
- 结构体声明
- 联合体声明
- 位域
- _Alignas
- typedef 声明
- 原子类型
- 外部及试探性定义
- inline 函数指定符
- _Noreturn 函数指定符
- 变长参数
- 内联汇编
- 可分析性
- 替用运算符及记号
- C 关键词
- 预处理器
- C 标准库头文件
- 类型支持
- 程序支持工具
- 变参数函数
- 错误处理
- 动态内存管理
- 日期和时间工具
- 字符串库
- 算法
- 数值
- 文件输入/输出
- 本地化支持
- 原子操作库
- 线程支持库
- 实验性 C 标准库
- 有用的资源
- 符号索引
- 注释
原子类型
语法
_Atomic ( type-name )
|
(1) | (C11 起) | |||||||
_Atomic type-name
|
(2) | (C11 起) | |||||||
1) 用作类型指定符;指代新的原子类型
2) 用作类型限定符;指代 type-name 的原子版本。在此作用中,它可以与 const 、 volatile 及 restrict 混合使用。尽管不同于其他限定符, type-name 的原子版本可能拥有不同的大小、对齐以及对象表示。
type-name | - | 任何异于数组或函数的类型。对于 (1) , type-name 亦不能为原子或 cvr 限定 |
头文件 <stdatomic.h>
定义 37 个便利宏,从 atomic_bool 到 atomic_uintmax_t ,它们简化此关键词和内建及库类型的一同使用。
_Atomic const int * p1; // p 是指向 _Atomic const int 的指针 const atomic_int * p2; // 同上 const _Atomic(int) * p3; // 同上
解释
原子类型的对象是仅有的免除数据竞争的对象,即它们可以被两个线程共时修改,或先被一个修改再被另一个读取。.
每个原子对象都拥有关联于其自身的修改顺序,即对该对象的完整修改顺序。若从某个线程的视角来看,对于某原子对象M的修改 A
发生先于同一原子对象 M 的修改 B
,则在 M 的修改顺序中 A
的出现先于 B
。
注意即使每个原子对象都有其自身的修改顺序,它却不是全序;不同线程可能会观测到相异原子对象有相异的修改顺序。
对于所有原子运算,保证有四种连贯:
- 写写连贯:若原子对象 M 的修改操作 A 先发生于 M 的修改操作 B ,则 M 的修改顺序中 A 出现早于 B 。
- 读读连贯:若原子对象 M 的值计算 A 先发生于 M 的值计算 B ,且从 M 上的副效应 X 求得 A 值,则 B 所计算得的值要么是 X 所存储的值,要么是 M 上的副效应 Y 所存储的值,其中 Y 在 M 的修改顺序中出现后于 X 。
- 读写连贯:若原子对象 M 的值计算 A 先发生于 M 上的操作 B ,则从 M 上的副效应 X 求得 A 值,这里 X 在 M 的修改顺序中出现先于 B 。
- 写读连贯:若在原子对象 M 上的副效应 X 先发生于 M 的值计算 B ,则求值 B 从 X,或从在 M 的修改顺序中出现后于 X 的副效应 Y 求得其值。
一些原子运算亦是同步操作:它们可以拥有附加的释放语义、获取语义,或顺序一致语义。见 memory_order 。
内建的自增减运算符和复合赋值运算符是拥有完全序列一致顺序(如同用 memory_order_seq_cst )的读-修改-写操作。若想要较不严格的同步语义,则可以用标准库函数替代。
原子属性仅对左值表达式有意义。左值到右值转换(模仿从原子区域到 CPU 寄存器的内存读取)会把原子性及其他限定符剥去。
本节未完成 原因:更多和 memory_order 及原子库页面的综述互动 |
注意
若编译器定义了宏常量 __STDC_NO_ATOMICS__
(C11) ,则不提供关键词 _Atomic 以及 <stdatomic.h>
。
访问原子结构体/联合体的成员是未定义行为。
库类型 sig_atomic_t 不提供线程间同步或内存顺序,仅提供原子性。
volatile 类型不提供线程间同步、内存顺序或原子性。
关键词
示例
运行此代码
#include <stdio.h> #include <threads.h> #include <stdatomic.h> atomic_int acnt; int cnt; int f(void* thr_data) { for(int n = 0; n < 1000; ++n) { ++cnt; ++acnt; // 对于此例,宽松内存顺序是足够的,例如 // atomic_fetch_add_explicit(&acnt, 1, memory_order_relaxed); } return 0; } int main(void) { thrd_t thr[10]; for(int n = 0; n < 10; ++n) thrd_create(&thr[n], f, NULL); for(int n = 0; n < 10; ++n) thrd_join(thr[n], NULL); printf("The atomic counter is %u\n", acnt); printf("The non-atomic counter is %u\n", cnt); }
可能的输出:
The atomic counter is 10000 The non-atomic counter is 8644
引用
- C11 standard (ISO/IEC 9899:2011):
- 6.7.2.4 Atomic type specifiers (p: 121)
- 7.17 Atomics <stdatomic.h> (p: 273-286)