atomic用法
memory order
源码变成可执行程序,一般由预编译,编译,汇编,链接。源码重排序一般分为编译期重排序和运行期重排序。
编译期重排序:编译器在不改变单线程程序的语义的前提下,可以重新安排语句的执行顺序。在不改变程序的语义的前提下,尽可能减少寄存器的读取,存储次数,充分复用寄存器的存储值。
CPU乱序执行
| 名称 | 语义 |
| memory_order_relaxed | Relaxed语义 |
| memory_order_consume | Release-Acquire语义 |
| memory_order_acquire | Release-Acquire语义 |
| memory_order_release | Release-Acquire语义 |
| memory_order_acq_rel | Release-Acquire语义 |
| memory_order_seq_cst | Sequential consistency语义 |
synchronizes-with、happens-before关系
synchronizes-with: 如果线程A存储一个值,而线程B读取该值,那么线程A中的存储和线程B的载入之间存在一种synchronizes-with关系
happens-before: 如果一个操作排在另外一个操作之前,那么该操作就该发生于另一个操作之前
Relaxed语义
最宽松的内存操作约定,不会保证修改会不会及时被其他的线程看到,也不对乱序做任何要求
relaxed的原子类型操作不参与synchronizes-with关系。
不同变量的relaxed可以被自由的重排,前提它们服从所有约束下的happens-before关系
#include <atomic>
#include <thread>
#include <assert.h> atomic<bool> x, y;
atomic<int> z; void write_x_then_y()
{
x.store(true, memory_order_relaxed);
y.store(true, memory_order_relaxed);
} void read_y_then_x()
{
while (!y.load(memory_order_relaxed));
if (x.load(memory_order_relaxed))
++z;
}int main()
{
x = false;
y = false;
z = ;
thread a(write_x_then_y);
thread b(read_y_then_x);
a.join();
b.join();
assert(z.load() != ); // z可能等于0
}
Release-Acquire语义Sequential consistency语义
release和acquire总是一起使用
release用于写操作,acquire用于读操作
release之前的写操作不允许乱序到release之后, acquire之前的读操作不允许乱序到acquire之前
acquire的修改会及时被release看到
Sequential consistency语义
sequential consistency 相当于 release + acquire 之外,还加上了一个对该操作加上全局顺序的要求
#include <atomic>
#include <thread>
#include <assert.h> atomic<bool> x, y;
atomic<int> z; void write_x()
{
x.store(true, memory_order_seq_cst);
} void write_y()
{
y.store(true, memory_order_seq_cst);
} void read_x_then_y()
{
while (!x.load(memory_order_seq_cst));
if (y.load(memory_order_seq_cst))
++z;
} void read_y_then_x()
{
while(!y.load(memory_order_seq_cst));
if (x.load(memory_order_seq_cst))
++z;
} int main()
{
x = false;
y = false;
z = ;
thread a(write_x);
thread b(write_y);
thread c(read_x_then_y);
thread d(read_y_then_x);
a.join();
b.join();
c.join();
d.join();
assert(z.load() != ); // z不可能等于0
}
store,load
atomic<bool> x, y;
atomic<int> z; void write()
{
x.store(true, memory_order_relaxed);
y.store(true, memory_order_release); // x的值比y先填充值
} void read()
{
while(!y.load(memory_order_acquire)); // y在等write值得填充
if (x.load(memory_order_relaxed)) {
z++;
}
} int main()
{
x = false;
y = false;
z = ; thread t1(write);
thread t2(read);
t1.join();
t2.join(); cout << "z: " << z << endl;
}
exchage
compare_exchange_weak
bool compare_exchange_weak(T& expected, T val, memory_order sync = memory_order_seq_cst)
NOTE:
1. atomic变量的值与expected进行比较, 如果结果true, 用val更新atomic的值(like store); 如果结果为false, 用atomic的值更新expected的值.
2. 这个函数能获得这个atomic变量的值,并且如果比较的结果是true的话就修改这个值. 整个操作是原子操作,在读取或者修改这个值得瞬间,其他的线程不会修改这个值.
3. memory_order 是否生效也是根据比较的结果,如果结果为true,那么生效,否则不生效
4. compare_exchage_weak允许伪失败(fail spuriously),尽管expected值确实和atomic变量的值相等,仍然会返回false; 这需要使用while操作
#include <iostream>
#include <atomic>
#include <thread>
#include <vector>
using namespace std; struct Node {
int value;
Node *next;
}; atomic<Node*> list_head(nullptr);
void Append(int val)
{
Node *p_old_node = list_head;
Node *p_new_node = new Node {val, p_old_node}; while (!list_head.compare_exchange_weak(p_old_node, p_new_node)) {
p_new_node->next = p_old_node;
}
} int main()
{
vector<thread> threads;
for (int i = ; i < ; i++) {
threads.push_back(thread(Append, i));
} for (auto &i : threads) {
i.join();
} for (Node *it = list_head; it != nullptr; it = it->next) {
cout << it->value << " ";
}
cout << endl;
}
compare_exchange_strong
bool compare_exchange_strong (T& expected, T val, memory_order sync = memory_order_seq_cst)
1. compare_exchange_strong的用法和compare_exchange_weak基本用法都一样
2. compare_exchange_strong不允许伪失败(fail spuriously)
3. compare_exchange_weak的循环结构在某些机器上可能有更好的性能
atomic<int> ai;
int tst_val = ;
int new_val = ;
bool exchanged = false; void valsout()
{
cout << "ai: " << ai << " tst_val: " << tst_val << " new_val: " << new_val << " exchanged: " << boolalpha << exchanged << endl;
} int main()
{
ai = ;
valsout(); //ai = 3, tst_val = 4, new_val = 5, exchanged = false; exchanged = ai.compare_exchange_strong(tst_val, new_val);
valsout(); //ai = 3, tst_val = 3, new_val = 5, exchanged = false; exchanged = ai.compare_exchange_strong(tst_val, new_val);
valsout(); //ai = 3, tst_val = 3, new_val = 5, exchanged = false;
}
参考资料:
[1] http://www.cplusplus.com/reference/atomic/atomic/
[2] <<c++并发编程>>
[3] http://www.cnblogs.com/haippy/p/3252056.html
[4] http://www.cnblogs.com/muhe221/articles/5049474.html
atomic用法的更多相关文章
- boost并发编程boost::atomic
三个用于并发编程的组件: atomic,thread,asio(用于同步和异步io操作) atomic atomic,封装了不同计算机硬件的底层操作原语,提供了跨平台的原子操作功能,解决并发竞争读 ...
- C ++ _多线程笔记
#include<iostream>#include <thread>//创建线程需要添加的头文件 using namespace std;/*thread join(阻塞:主 ...
- Java:多线程,java.util.concurrent.atomic包之AtomicInteger/AtomicLong用法
1. 背景 java.util.concurrent.atomic这个包是非常实用,解决了我们以前自己写一个同步方法来实现类似于自增长字段的问题. 在Java语言中,增量操作符(++)不是原子的,也就 ...
- atomic 原子自增工程用法案例
案例 1 : 简单用法 atomic_int id; atomic_fetch_add(&id, 1) atomic_uint id; atomic_fetch_add(&id, 1) ...
- 原子操作--sync/atomic的用法
golang 通过sync/atomic库来支持cpu和操作系统级别的原子操作.但是对要操作类型有如下要求 int32, int64,uint32, uint64,uintptr,unsafe包中的P ...
- 多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类)
前言:刚学习了一段机器学习,最近需要重构一个java项目,又赶过来看java.大多是线程代码,没办法,那时候总觉得多线程是个很难的部分很少用到,所以一直没下决定去啃,那些年留下的坑,总是得自己跳进去填 ...
- java并发:线程同步机制之Volatile关键字&原子操作Atomic
volatile关键字 volatile是一个特殊的修饰符,只有成员变量才能使用它,与Synchronized及ReentrantLock等提供的互斥相比,Synchronized保证了Synchro ...
- C++11 并发指南六(atomic 类型详解三 std::atomic (续))
C++11 并发指南六( <atomic> 类型详解二 std::atomic ) 介绍了基本的原子类型 std::atomic 的用法,本节我会给大家介绍C++11 标准库中的 std: ...
- nonatomic, retain,weak,strong用法详解
strong weak strong与weak是由ARC新引入的对象变量属性 ARC引入了新的对象的新生命周期限定,即零弱引用.如果零弱引用指向的对象被deallocated的话,零弱引用的对象会被自 ...
随机推荐
- C++类对象之间的类型转换和重载
类对象和其他类型对象的转换 转换场合有: 赋值转换 表达式中的转换 显式转换 函数调用, 传递参数时的转换 转换方向有: 由定义类向其他类型的转换 由其他类型向定义类的转换 #include < ...
- HBase之四--(3):hbasehbase分页查询
为了广大技术爱好者学习netty,在这里帮新浪微博@nettying宣传下他出版的新书 <netty权威指南>@nettying兄在华为NIO实践多年,这本书是他的技术和经验的一个结晶.N ...
- JDK1.8中对hashmap的优化
在Java编程语言中,最基本的结构就是两种,一个是数组,另外一个是模拟指针(引用),所有的数据结构都可以用这两个基本结构来构造的,HashMap也不例外.HashMap实际上是一个“链表散列”的数据结 ...
- Flutter实战视频-移动电商-52.购物车_数据模型建立和Provide修改
52.购物车_数据模型建立和Provide修改 根据json数据生成模型类 {,"price":830.0,"images":"http://imag ...
- 转载 关于启用HTTPS的一些经验分享
本文转载自 https://imququ.com/post/sth-about-switch-to-https.html 随着国内网络环境的持续恶化,各种篡改和劫持层出不穷,越来越多的网站选择了全站 ...
- CCF 201509-3 模板生成系统 (STL+模拟)
问题描述 成成最近在搭建一个网站,其中一些页面的部分内容来自数据库中不同的数据记录,但是页面的基本结构是相同的.例如,对于展示用户信息的页面,当用户为 Tom 时,网页的源代码是 而当用户为 Jerr ...
- jQuery 学习笔记(一)jQuery 语法
jQuery 是一个 JavaScript 库,极大地简化了 JavaScript 编程,很容易学习 添加 jQuery 库 <head> <script type="te ...
- 细说CSS中的display属性
相信大部分奋战在前端的,尤其在前端攻城的过程中,有一种越陷越深的感觉,不错,一如前端深似海,从此妹子是浮云啊,前端上手容易,深入难啊!下面我就CSS中的display属性讲下我自己所积累的,与大家共享 ...
- E20180601-hm
trade-off n. 权衡; 交易;(不是商业方面的交易,而是“利”与“弊”的权衡) vertex n. 顶点; 最高点; <数>(三角形.圆锥体等与底相对的)顶; (三角形.多边形等 ...
- TP5之一次选择多张图片并预览
点击选择图片(可选多张),确定后将选择的图片显示在页面上,已经选择的图片也可以删除,点击提交将图片提交给后台. 1.效果图 2.code 用input标签并选择type=file,记得带上multip ...