虚拟主机 2个网站,风兰网络,深圳黄页企业名录,html5企业网站赏析1.原子操作 多线程下为了实现对临界区资源的互斥访问#xff0c;最普遍的方式是使用互斥锁保护临界区。 然而#xff0c;如果临界区资源仅仅是数值类型时#xff0c;对这些类型c提供了原子类型#xff0c;通过使用原子类型可以更简洁的获得互斥保护的支持。
(1). 一个实例…1.原子操作 多线程下为了实现对临界区资源的互斥访问最普遍的方式是使用互斥锁保护临界区。 然而如果临界区资源仅仅是数值类型时对这些类型c提供了原子类型通过使用原子类型可以更简洁的获得互斥保护的支持。
(1). 一个实例
#include atomic
#include thread
#include iostream
using namespace std;atomic_llong total{0};
void func(int){for(long long i 0; i 100000; i){total i;}
}int main(){thread t1(func, 0);thread t2(func, 0);t1.join();t2.join();cout total endl;return 0;
}上述由于使用了atomic_llong类型的原子变量所以 total i;操作是具备互斥保护的。
(2). cstdatomic中的原子类型和内置类型对应表
原子类型名称对应的内置类型名称atomic_boolboolatomic_charcharatomic_scharsigned charatomic_ucharunsigned charatomic_intintatomic_uintunsigned intatomic_shortshortatomic_ushortunsigned shortatomic_longlongatomic_ulongunsigned longatomic_llonglong longatomic_ullongunsigned long longatomic_char16_tchar16_tatomic_char32_tchar32_tatomic_wchar_twchar_t
(3).另一种使用原子类型的方式 使用std::atomicT模板类。 注意点 a.该模板类不支持拷贝构造移动构造赋值运算符。 b.std::atomicT定义了到T的类型转换函数。
(4).atomic类型及其相关的操作
操作atomic_flagatomic_boolatomic_integral_typeatomicboolatomicT*atomicintegral-typeAtomicclass-typetest_and_setYclearYis_lock_freeyyyyyyloadyyyyyystoreyyyyyyexchangeyyyyyycompare_exchange_weakstrongyyyyyyfetch_add,yyyfetch_sub,-yyyfetch_or,|yyfetch_and,yyfetch_xor,^yy,--yyyy
(5).使用atomic_flag可自行实现自旋锁
#include thread
#include atomic
#include iostream
#include unistd.h
using namespace std;std::atomic_flag lock ATOMIC_FLAG_INIT;
void f(int n){while(lock.test_and_set())cout waiting from thread n endl;cout thread n starts working endl;
}void g(int n){cout thread n is going to start. endl;lock.clear();cout thread n starts working endl;
}int main(){lock.test_and_set();thread t1(f, 1);thread t2(g, 2);t1.join();usleep(100);t2.join();return 0;
}2.顺序一致性内存模型 默认下使用原子类型时自然就是顺序一致的。即指令实际被cpu执行的顺序和高级语言中书写顺序是一致的。 有时对某些并发场景我们可能并不需要如此严格的限制也能保证指令执行的正确性我们可以借助顺序一致性内存模型的显式控制来达到此目的。
一个实例
#include thread
#include atomic
#include iostream
using namespace std;atomicint a{0};
atomicint b{0};
int ValueSet(int){int t 1;a t;b 2;
}int Observer(int){cout ( a , b ) endl;
}int main(){thread t1(ValueSet, 0);thread t2(Observer, 0);t1.join();t2.join();cout Got ( a , b ) endl;return 0;
}上述实例中线程t1依次对ab执行赋值。线程t2依次读取ab的值。 但从高级语言到处理器执行二进制指令的实际效果并不一定严格按上述预期的顺序来。 从高级语言到处理器执行二进制指令有两个阶段会影响指令实际执行的顺序
(1). 编译阶段 编译器处于性能优化考虑针对没有执行依赖的语句可能生成汇编代码时调整指令顺序。 上述实例在执行汇编时线程t1中 int t 1;a t;和b 2;没有依赖关系所以允许安排汇编语句时b 2;对应的汇编语句在int t 1;a t;对应的汇编语句之前或中间。线程t2中访问a访问b类似。
顺序一致性指的是编译后的汇编指令顺序和高级语言中顺序是否一致。 (2).二进制指令执行阶段 假设编译器按高级语言一致顺序产生了如下汇编代码
1 Loadi reg3, 1; #将立即数1放入寄存器reg3
2 Move reg4, reg3; #将reg3的数据放入reg4
3 Store reg4, a; #将寄存器reg4中的数据存入内存地址a
4 Loadi reg5, 2; #将立即数2放入寄存器reg5
5 Store reg5, b; #将寄存器reg5中的数据存入内存地址b处理器实际执行二进制指令时由于上述123和45没有依赖关系所以某些cpu体系结构下45可能在123之前或123中间被执行。
这里我们称严格按二进制指令顺序执行指令的cpu体系结构为强顺序的反之则为弱顺序的。 所以内存模型是一个针对cpu体系结构的概念。
弱顺序体系结构下保证指令执行顺序符合预期的手段是添加额外的汇编指令。
1 Loadi reg3, 1; #将立即数1放入寄存器reg3
2 Move reg4, reg3; #将reg3的数据放入reg4
3 Store reg4, a; #将寄存器reg4中的数据存入内存地址a
Sync
4 Loadi reg5, 2; #将立即数2放入寄存器reg5
5 Store reg5, b; #将寄存器reg5中的数据存入内存地址b由于添加了额外的Sync汇编指令即使在弱内存cpu体系结构下执行上述汇编指令也能保证先执行123再执行45。 像Sync这样的汇编指令称为内存栅栏。
3.高级语言如何保证指令执行顺序和预期代码中出现顺序一致 (1).编译阶段保证得到的汇编指令顺序和高级语言中一致。 (2).针对强顺序cpu体系结构无需额外处理。针对弱顺序cpu体系结构在汇编指令中额外插入内存栅栏。 默认情况下使用原子操作时上述(1)(2)均是满足的。
4.通过放松一致性要求来提高执行效率 c的原子操作大多都可以使用memory_order作为一个参数。 c11中memory_order所有可能取值
枚举值定义规则memory_order_relaxed不对执行顺序做任何保证memory_order_acquire本线程中所有后续读操作必须在本条原子操作完成后执行memory_order_release本线程中所有之前的写操作完成后才能执行本条原子操作memory_order_acq_relmemory_order_acquire memory_order_releasememory_order_consume本线程中所有后续的有关本原子类型的操作必须在本条原子操作完成后执行memory_order_seq_cst全部存取操作都按顺序执行
memory_order_seq_cst 是c11所有原子操作默认值。具备最强一致性要求。 通常可把atomic的成员函数可使用的memory_order分为三组 (1). 原子存储操作store memory_order_relaxed 、memory_order_release 、memory_order_seq_cst (2).原子读取操作load memory_order_relaxed、memory_order_consume、memory_order_acquire 、memory_order_seq_cst (3).同时读写操作 全部六种
5.利用显式设置memory_order保证原子操作既快又对的实例 5.1.默认版本
#include thread
#include atomic
#include iostream
using namespace std;atomicint a;
atomicint b;
int Thread1(int){int t 1;a t;b 2;
}void Thread2(int){while(b ! 2);cout a endl;
}int main(){thread t1(Thread1, 0);thread t2(Thread2, 0);t1.join();t2.join();return 0;
}上述t2种预期打印出来的a应该是1。 原子操作默认下会保证严格的顺序一致性编译层面cpu体系执行层面若我们希望维持预期下放松一致性要求就需要通过显式设置memory_order来达到目的。
5.2.一个一致性要求略低但保证符合预期的版本
#include thread
#include atomic
#include iostream
using namespace std;atomicint a;
atomicint b;
int Thread1(int){int t 1;a.store(t, memory_order_relaxed);b.store(2, memory_order_release);
}int Thread2(int){while(b.load(memory_order_acquire) ! 2);cout a.load(memory_order_relaxed) endl;
}int main(){thread t1(Thread1, 0);thread t2(Thread2, 0);t1.join();t2.join();return 0;
}t1中memory_order_release会保证a的写入先执行再执行b的写入。 t2中memory_order_acquire会保证先读取b再读取a。 上述两个限制下我们知道t2中a将会符合预期。