上一篇讲述了 如何去管理线程,std::thread 线程的创建、等待、分离、共享、所有权的转移,基础的理念,这一篇准备了解一下线程间数据共享机制,有以下模块。

一、线程数据共享

在同一个进程当中,允许多线程的存在,不管是不是有无数据共享,多线程之间仍然保持着资源竞争,共享同一个进程的资源,并无时无刻的切换上下文进行cpu 的调度,如当访问同一个变量的时候,原因对变量的操作,是先从内存中读取数据到寄存器当中,修改完成之后,在重新写入内存,由于这期间线程先后访问不一,可以对同一变量同时进行读写操作,造成变量的不确定性,因此如果不控制对共享数据的访问先后,会产生错误的结果看,这样就会带来问题。

#include <thread>
#include <iostream>int i=0;void test(){i++;std::cout<<"i value:"<<i<<std::endl;
}int main(int argc, char *argv[])
{int cpu_num = std::thread::hardware_concurrency();std::cout<<"cpu_num value:"<<cpu_num<<std::endl;std::thread threads[cpu_num];for(int n=0;n<cpu_num;n++){threads[n] = std::thread (test);}for (auto& t: threads) {t.join();}
}

以上代码就会造成对共享资源 i 最后结果的不一致性,这就是多线程访问会造成的影响 。

二、使用互斥量访问共享资源

因为多线程之间的竞争导致结果的不一致性,我们可以使用互斥量进行访问,如下代码简述

#include <thread>
#include <iostream>
#include <mutex>std::mutex mu;int i=0;void test(){std::lock_guard<std::mutex> guard(mu);i++;std::cout<<"i value:"<<i<<std::endl;
}int main(int argc, char *argv[])
{int cpu_num = std::thread::hardware_concurrency();std::cout<<"cpu_num value:"<<cpu_num<<std::endl;std::thread threads[cpu_num];for(int n=0;n<cpu_num;n++){threads[n] = std::thread (test);}for (auto& t: threads) {t.join();}
}

通常不直接使用std::mutex的成员函数lock 借鉴于RALL 思想,在函数执行完毕的时候进行解琐,所以会使用  std::lock_guard<std::mutex> guard(mu);作为一种处理方式。

三、保护共享数据的方式

方式1、一些资源仅需要在第一次初始化的时候需要保护,其时候就可以不需要互斥变量的保护了。比如编码中最常见的单例模式,核心代码如下.,创建单例是一种保护方式。

//(3)获得本类实例的唯一全局访问点
static CSinglton* GetInstance()
{//若实例不存在,则创建实例对象if (NULL == pInstance){pInstance = new CSinglton();}
//实例已经存在,直接该实例对象return pInstance;
}

方式2、使用双重锁定方式单例模式,主要是使用互斥量来进行初始化。

static CSinglton* GetInstance()
{//仅在实例未被创建时加锁,其他时候直接返回if (NULL == pInstance){std::lock_guard<std::mutex> guard(g_mutex)if (NULL == pInstance){//若实例不存在,则创建实例对象pInstance = new CSinglton();//步骤一}}//实例已经存在,直接该实例对象return pInstance;
}
//执行代码
GetInstance()->dosomething();//步骤二

方式3、std::call_once保护共享数据
为了解决双重锁定以及创建多个实例的问题,C++11标准库提供了std::once_flag和std::call_once来处理只初始化一次的情况。使用std::call_once比显式使用互斥量消耗的资源更少,并且还是无锁式编程,减少了死锁问题的发生。

std::once_flag flag;void simple_do_once()
{std::call_once(flag, [](){ std::cout << "Simple example: called once\n"; });
}int _tmain(int argc, _TCHAR* argv[])
{std::thread st1(simple_do_once);std::thread st2(simple_do_once);std::thread st3(simple_do_once);std::thread st4(simple_do_once);st1.join();st2.join();st3.join();st4.join();std::cout << "main thread end\n";
}

由于本人经验有限,如有错误,欢迎修正。

c++基础11、线程间数据共享相关推荐

  1. python线程间数据共享_python 进程间数据共享multiProcess.Manger实现解析

    一.进程之间的数据共享 展望未来,基于消息传递的并发编程是大势所趋 即便是使用线程,推荐做法也是将程序设计为大量独立的线程集合,通过消息队列交换数据. 这样极大地减少了对使用锁定和其他同步手段的需求, ...

  2. 保护线程间的共享数据

    程序员的自我修养(六):保护线程间的共享数据 多进程和多线程最本质的区别在于共享和隔离的程度不同.对于多进程方式来说,因为隔离程度高,所以程序员很少需要去担心进程空间的数据被破坏:但是并发任务之间共享 ...

  3. rtth之线程间的通信

    IPC基础 线程间同步 一.前言 除了抢占式调度机制,各线程间同步是相互协作处理事件和数据的基础. 线程间通信/同步(IPC:Inter-Process Communication)是一个多任务操作系 ...

  4. C++11 线程通信的三种方式

    前言:发出消息的线程叫做检测线程,响应消息的线程称为响应线程. C++11线程间的通信方式有三种: (1)通过条件变量进行线程间的通信 (2)通过标志位来通知线程间的通信 (3)通过std::furt ...

  5. Linux 线程间通信方式、进程通信方式

    Linux线程间通信几种主要的手段 1. 管道:         管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有 ...

  6. 面试题:进程间通信方式,线程间通信方式

    一.进程间通信(IPC,Inter-Process Communication)是指在不同进程间传播或交换信息 1. 无名管道 特点 半双工(数据流向仅有一个方向),具有固定的读端和写端 只能用于父进 ...

  7. java线程通信概念_java基础线程总结(线程概念、线程创建方式、线程间通信、线程重要方法)...

    基础篇之<线程> @author :kern ---------------------------------------------------------------- 一:进程:是 ...

  8. 线程间通讯机制(基础篇)——Handler、Runnable、HandlerThread、AsyncTask的使用

    前言: android线程通讯机制是android应用开发的基础课程,对于很多初学android的朋友可能还没有完全理解,所以,今天我就做一下知识小结吧. 一.线程安全 可能有java基本的朋友都知道 ...

  9. Java基础学习——多线程(线程间通信-生产者消费者代码示例)

    JDK 1.5提供了多线程升级方案 将同步synchronized替换成了显示的Lock操作.可以实现唤醒.冻结指定的线程. Lock接口 Lock 实现提供了比使用 synchronized 方法和 ...

  10. java基础—线程间的通讯 生产者与消费者

    线程间的的通讯  生产者与消费者 public class TestDemos3 {public static void main(String[] args){Res r = new Res();I ...

最新文章

  1. leetcode算法题--三角形最小路径和
  2. HTTP请求分析工具Fiddler
  3. html5点击显示展开列表,HTML5 - 如何折叠和展开复杂的表格元素
  4. c++新特性11 (11)unique_ptr
  5. containers文件夹可以删除吗_iOS14描述文件可以删除吗?iPhone删除iOS14描述文件方法...
  6. Linux——安装之磁盘分区
  7. C++学习系列笔记(九)
  8. linux losetup
  9. 【贪心】hdu5969 最大的位或
  10. Java基础---Java---面试题---交通灯管理系统(面向对象、枚举)
  11. 笔记本电脑风扇转速测试软件,笔记本温度查看和风扇转速调节工具
  12. Windows端口扫描关闭工具
  13. Stable Baselines/用户向导/矢量化环境
  14. 只有1kb的清理软件_1kb病毒专杀工具下载
  15. STL之容器deque
  16. xcode 软件˙∆集~
  17. JAVA 系列——包装类
  18. iOS 动态加载LaunchScreen上的图片
  19. Flash页面的在线编辑
  20. ArcMap:线、面符号的制作(通用线符号的制作)超详细

热门文章

  1. 通过NCNN在树莓派上部署Fastest YOLO
  2. 安卓右侧字母索引,带滑动气泡,仿微信
  3. 新生干货!--开学季,我想对大一大二同学说:
  4. 北邮计算机发展,准备报考北京邮电大学的信息科学与技术专业,发展前景如何?...
  5. ftp connect 连接被拒绝
  6. 小小知识点(三十一)MU-MIMO和SU-MIMO分别表示什么?
  7. python气象插值_Python气象数据处理进阶之Xarray(3):插值
  8. oracle 报错2PC,关于ORACLE的ora-12505报错以及连接问题的解决及相关资料 (2)
  9. 修改Swagger接口文档网络请求返回状态码
  10. 电影推荐之《 哈利波特与密室》 隐私策略(Privacy policy)