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的话,零弱引用的对象会被自 ...
随机推荐
- mysql的navicat注册码生成
首先下载安装Navicat在Navicat关闭的情况下运行注册机在注册机界面点击patch,选择Navicat安装目录下的Navicat.exe打补丁弹出破解成功后拔掉网线断网products选择my ...
- 通用方法解决UITextFiled输入的时候,键盘遮挡问题
我们在用键盘录入的时候,有可能会遮挡录入框,所以我们应调整UIView的位置,使其不被遮挡.我写了一个通用的方法可以解决这个问题:1. [代码][C/C++]代码 - (void) ...
- kafka之六:为什么Kafka那么快
转自: http://mp.weixin.qq.com/s?__biz=MzIxMjAzMDA1MQ==&mid=2648945468&idx=1&sn=b622788361 ...
- js数组,在遍历中删除元素(用 for (var i in arr)是无效的 )
/** * 有效的方式 - 改变下标,控制遍历 */ for (var i = 0; i < arr.length; i++) { if (...) { arr.splice(i, 1); // ...
- CreateEx
virtual BOOL CreateEx( DWORD dwExStyle, LPCTSTR lpszClassName, LPCTSTR lpszWindowName, DWORD dwStyle ...
- Numbers Exchange
题意: Eugeny有n张卡片,他希望和Nikolay交换一些卡片使得他拥有的奇数数字和偶数数字的卡片数目一样,且所有数字都不同. Nikolay有m张卡片,分别写着1到m.问最少交换几次,能够满足要 ...
- mysql分页性能
- select * from userinfo limit 20000,10 # 数据越往后越慢 - 索引表中扫: select * from userinfo where id in (sele ...
- 4-1逻辑与运算符介绍 & 4-2逻辑或运算符介绍
后面括号内的(n++)不运算了. 4-2逻辑或运算符介绍
- E20180519-hm
distinct adj. 明显的,清楚的; 卓越的,不寻常的; 有区别的; 确切的;
- spring基于注解的IOC
曾经的XML配置: <bean id="accountService" class="com.itheima.service.impl.AccountService ...