线程安全几个重要概念


实现线程同步的几种手段:
     二元信号量:
          一种最简单的锁,只有两种状态 占用和非占用 适合只能被一个线程独占访问的资源
     多元信号量(信号量):
          一种初始化为N的信号量 允许最多N个线程对资源进行同时访问
     互斥量:
          和二元信号量类似,但是它只能用于实现对某个资源的独立访问 即不可用于实现同步 因为哪个线程获取了互斥量 该线程就必须释放该互斥量 而信号量可以由其他线程释放 用以实现同步
     临界区:
          比互斥量更加严格的一种互斥手段,因为它只在本进程可见 而互斥量在所有进程中可见 其他与互斥量相同
     读写锁:
          有三种状态 自由 共享和独占  它们各自被线程访问的权限不同 用于灵活的满足那些读取次数多 写入次数少的资源访问控制 在线程获得访问权限后 可以修改其读写锁状态  获取方式分为共享方式和独占方式
          自由:     两种获取方式均可获取权限
          共享:    仅共享方式可获取权限
          独占      只能等待当前使用线程使用完毕 并且修改读写锁状态才可获取
     条件变量:
          同时注册一个事件,让线程等待某一个事件触发条件变量来唤醒等待线程 该事件在某时刻被另一个线程触发 一个条件变量可以同时被多个线程等待 当条件变量有效后 所有线程得以恢复执行
     

 可重入函数:
     一个函数可重入,是指这个函数还没有执行完成,由于外部因素或内部调用,又一次进入该函数进行 主要有两种情况
     1.多个线程同时执行这个函数
     2.函数直接或间接地调用自身
     一个函数被称为是可重入的。表面该函数被重入之后不会产生任何不良后果。这需要函数满足一下几个条件
  •    不使用任何(局部)静态或者全局的非const变量
  •    不返回任何(局部)静态或者全局的非const变量的指针
  •    仅依赖于调用方提供的参数
  •    不依赖任何单个资源的锁(mutex等)
   不调用任何不可重入的函数

     如果一个函数满足以上条件 那么它是可重入的 可以安全的在多线程环境中使用

编译器的过度优化与CPU动态调度:
     由于现代编译器的过度优化,使得有时候我们即使合理的使用了锁也不一定能保证线程安全
     eg1
     x = 0;
     thread1       thread2
     lock();       lock();
     x++;          x++
     unlock();     unlock();
     上面两个线程执行完之后,x的值可不一定为2
     想象如下操作流程
     1.thread1 读取x到寄存器A 此时[A]=0;
     2.thread1 修改x  x++  此时[A] = 1;
     3.thread1 延迟写回缓存 (编译器优化)
     4.thread2 读取x到寄存器B(注意,在thread2中 它并不知道x的值放在寄存器A中 不同线程有着不同的寄存器环境)此时[B] = 0;
     5.thread2 修改x x++ 此时[B] = 1;
     6.接下来thread1 thread无论以何种顺序写回,x得到最终值为1
     eg2
     x = y = 0;
     thread1        thread2
     x = 1;         y = 1;
     r1 = y;        r2 = x;
     这样看起来,无论怎么执行 r1和r2必定至少有一个为1 然而未必
     由于CPU的动态调度,使得CPU在执行指令时为了提高效率可能会交换指令的顺序。另外 在编译器优化的时候,可能为了效率而交换毫不相干的两条相邻指令 也就是说 可能最终执行序列如下
     x = y = 0;
     thread1      thread2
     r1 = y;       y = 1;
     x = 1;        r2 = x;
     最终我们将得到r1 = r2 = 0
     对于上面两个问题 我们可以用volatile关键字实现阻止编译器过度优化 它可以做到:
     1.阻止编译器为了提高速度将一个变量缓存到寄存器内而不写回
     2.阻止编译器调整操作volatile变量的指令序列
     可见volatile可以解决第一个问题,但是volatile却不能解决第二个问题,因为即使阻止了编译器的动态调度,也无法阻止CPU的动态调度 
     这可以通过 barrier指令
     #define barrier() _asm_volatile ("lwsync")  //这里barrier()实现分隔其两边的指令流 不允许越界交换指令
     x = y = 0;
     thread1        thread2
     x = 1;          y = 1;
     barrier()      barrier();
     r1 = y;        r2 = x;
  
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值