C++ 多线程detach()操作的坑以及传参

2023-05-16

detach()的作用是将子线程和主线程的关联分离,也就是说detach()后子线程在后台独立继续运行,主线程无法再取得子线程的控制权即使主线程结束,子线程未执行也不会结束。当主线程结束时,由运行时库负责清理与子线程相关的资源。实际应用如让一个文字处理应用同时编辑多个文档,让每个文档处理窗口拥有自己的线程,每个线程运行同样的代码,并隔离不同窗口处理的数据。


 1 #include <thread>
 2 #include <iostream>
 3 
 4 using namespace std;
 5 
 6 void func()
 7 {
 8     cout << "子线程func开始执行!" << endl;
 9     //do something
10     cout << "子线程func执行结束!" << endl;
11 }
12 
13 int main()
14 {
15     cout << "主线程main开始执行!" << endl;
16     thread t(func);
17     t.detach();
18     cout << "主线程main执行结束!" << endl;
19     return 0;
20 }  

注意

  detach()同时也带来了一些问题,如子线程要访问主线中的对象,而主线中的对象又因为主线程结束而被销毁时,会导致程序崩溃。所以传递参数时需要注意一些陷阱。关于参数传递: 

  1、访问主线程对象以及指针问题

  2、构造线程时隐式转换问题,子线程可以还来不及转换,主线程对象就销毁了,解决方法是构造线程时,构造一个临时对象传入

线程可以共享进程的内存空间,线程拥有自己独立内存。

  关于参数的传递,std::thread的构造函数只会单纯的复制传入的变量,特别需要注意的是传递引用时,传入的是值的副本,也就是说子线程中的修改影响不了主线程中的值。

 

值传递

  主线程中的值,被拷贝一份传到了子线程中。


 1 #include <iostream>
 2 #include <thread>
 3 
 4 using namespace std;
 5 
 6 void test(int ti, int tj)
 7 {
 8     cout << "子线程开始" << endl;
 9     //ti的内存地址0x0055f69c {4},tj的内存地址0x0055f6a0 {5}
10     cout << ti << " " << tj << endl;  
11     cout << "子线程结束" << endl;
12     return;
13 }
14 
15 
16 int main()
17 {
18     cout << "主线程开始" << endl;
19     //i的内存地址0x001efdfc {4},j的内存地址0x001efdf0 {5}
20     int i = 4, j = 5;  
21     thread t(test, i, j);
22     t.join();
23     cout << "主线程结束!" << endl;
24     return 0;
25 }  

传引用

  从下面的运行结果,可以看出,即使是用引用来接收传的值,也是会将其拷贝一份到子线程的独立内存中,这一点与我们编写普通程序时不同。这是因为线程的创建属于函数式编程,所以为了传引用C++中才引入了std::ref()。关于std::ref()。


 1 #include <iostream>
 2 #include <thread>
 3 
 4 using namespace std;
 5 
 6 class A{
 7 public:
 8      int ai;
 9      A (int i): ai(i) { }
10 };
11 
12 //这种情况必须在引用前加const,否则会出错。目前本人的觉得可能是因为临时对象具有常性
13 void test(const int &ti, const A &t) 
14 {
15     cout << "子线程开始" << endl;
16     //ti的内存地址0x0126d2ec {4},t.ai的内存地址0x0126d2e8 {ai=5 }
17     cout << ti << " " << t.ai << endl;  
18     cout << "子线程结束" << endl;
19     return;
20 }
21 
22 
23 int main()
24 {
25     cout << "主线程开始" << endl;
26     //i的内存地址0x010ff834 {4},a的内存地址0x010ff828 {ai=5 }
27     int i = 4;  
28     A a = A(5);
29     thread t(test, i, a);
30     t.join();
31     cout << "主线程结束!" << endl;
32     return 0;
33 }  

   那么如果我们真的需要像一般程序那样传递引用呢,即在子线程中的修改能够反映到主线程中。此时需要使用std::ref()但是注意如果我们会在子线中改变它,此时用于接收ref()的那个参数前不能加const。关于C++多线程中的参数传引用问题,我目前只是记住了这个现象,关于原理还需后期研究源码继续学习。


 1 #include <iostream>
 2 #include <thread>
 3 
 4 using namespace std;
 5 
 6 class A {
 7 public:
 8     int ai;
 9     A(int i) : ai(i) { }
10 };
11 
12 //接收ref()的那个参数前不能加const,因为我们会改变那个值
13 void test(int& ti, const A& t)
14 {
15     cout << "子线程开始" << endl;
16     cout << ti << " " << t.ai << endl;
17     ti++;
18     cout << "子线程结束" << endl;
19     return;
20 }
21 
22 
23 int main()
24 {
25     cout << "主线程开始" << endl;
26     int i = 4;
27     A a = A(5);
28     thread t(test, ref(i), a);
29     t.join();
30     cout << "i改变:" << i << endl;
31     cout << "主线程结束!" << endl;
32     return 0;
33 }  

   传入类对象时,使用引用来接收比用值接收更高效。


 1 #include <iostream>
 2 #include <thread>
 3 
 4 using namespace std;
 5 
 6 class A {
 7 public:
 8     int ai;
 9     A (int i) : ai(i) 
10     { 
11         cout << "构造" << this << endl; 
12     }
13 
14     A (const A& a) :ai(a.ai) {
15         cout << "拷贝构造" << this << endl;
16     }
17 
18     ~A()
19     {
20         cout << "析构" << this << endl;
21     }
22 };
23 
24 //void test(const A a)
25 void test(const A& a)
26 {
27     cout << "子线程开始" << endl;
28     cout << "子线程结束" << endl;
29     return;
30 }
31 
32 
33 int main()
34 {
35     cout << "主线程开始" << endl;
36     int i = 4;
37     thread t(test, A(i));
38     t.join();
39     cout << "主线程结束!" << endl;
40     return 0;
41 }  

 

 

 

传指针

  从下面的运行结果,可以看出,主线程和子线程中的指针都是指向同一块内存。所以在这种情况下会有一个陷阱,如果使用detach(),则当主线程崩溃或者正常结束后,该块内存被回收,若此时子线程没有结束,那么子线程中指针的访问将未定义,程序会出错。


 1 #include <iostream>
 2 #include <thread>
 3 
 4 using namespace std;
 5 
 6 
 7 void test(char *p) 
 8 {
 9     cout << "子线程开始" << endl;
10     //0x004ffeb4 "hello"
11     cout << p << endl;  
12     cout << "子线程结束" << endl;
13     return;
14 }
15 
16 
17 int main()
18 {
19     cout << "主线程开始" << endl;
20     //0x004ffeb4 "hello"
21     char s[] = "hello";
22     thread t(test, s);
23     t.join();
24     cout << "主线程结束!" << endl;
25     return 0;
26 }  

 

传临时对象

   用临时变量作为实参时,会更高效,由于临时变量会隐式自动进行移动操作,这就减少了整体构造函数的调用次数。而一个命名变量的移动操作就需要std::move()。


 1 #include <iostream>
 2 #include <thread>
 3 
 4 using namespace std;
 5 
 6 class A {
 7 public:
 8     int ai;
 9     A (int i) : ai(i) 
10     { 
11         cout << "构造" << this << endl; 
12     }
13 
14     A (const A& a) :ai(a.ai) {
15         cout << "拷贝构造" << this << endl;
16     }
17 
18     ~A()
19     {
20         cout << "析构" << this << endl;
21     }
22 };
23 
24 void test(const A& a)
25 {
26     cout << "子线程开始" << endl;
27     cout << "子线程结束" << endl;
28     return;
29 }
30 
31 
32 int main()
33 {
34     cout << "主线程开始" << endl;
35     int i = 4;
36     thread t(test, A(i));
37     t.join();
38     cout << "主线程结束!" << endl;
39     return 0;
40 }  

总结

  1、使用引用和指针是要注意;

  2、对于内置简单类型,建议传值;

  3、对于类对象,建议使用引用来接收,以为使用引用会只会构造两次,而传值会构造三次;

  4、在detach下要避免隐式转换,因为此时子线程可能还来不及转换主线程就结束了,应该在构造线程时,用参数构造一个临时对象传入。

       5、实际上,也并不是说不能用detach(),有时候要在线程中和主线程共用某些地址,在我们前面的博客中,我们就在线程函数中传了指针,在线程函数中读取指针指向的位置的数据。但是,我们必须在主线程中的析构函数或者退出函数中将传出去的那些指针都赋为null,并在线程函数中对传进来的指针做出判断,判定是否为null,然后再执行相应的逻辑(潜在的风险是该地址在主线程中赋值为null,但是后面又起来一个程序使用了该地址并赋值)。或者加上flag控制开关,在主线程退出时将flag赋为false,在线程函数中获取flag然后执行逻辑。

后面我会写程序再验证上述观点。

参考出处:https://www.cnblogs.com/chen-cs/p/13056703.html

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C++ 多线程detach()操作的坑以及传参 的相关文章

  • Android登录注册功能封装

    我们都知道Android应用软件基本上都会用到登录注册功能 xff0c 那么对一个一个好的登录注册模块进行封装就势在必行了 这里给大家介绍一下我的第一个项目中所用到的登录注册功能的 xff0c 已经对其进行封装 xff0c 希望能对大家有帮
  • Kotlin 官方学习教程之扩展

    扩展 类似于 C 和 Gosu xff0c Kotlin 也提供了一种可以在不继承父类也不使用类似装饰器这样的设计模式的情况下对指定类进行扩展的功能 这是通过称为扩展名的特殊声明来实现的 Kotlin 支持函数扩展和属性扩展 函数扩展 要声
  • Kotlin 官方学习教程之密封类与泛型

    密封类 密封类用于表示受限类层次结构 xff0c 当值可以有一个有限集合的类型 xff0c 但不能有其他类型 它们在某种意义上是枚举类的扩展 xff1a 枚举类型的值集合也受到限制 xff0c 但每个枚举常量仅作为单个实例存在 xff0c
  • 致年轻时如此拼搏的你我

    离别总是伤人意 这一篇文章写在这个时候是有其特殊意义和价值 xff0c 起码对我来说是这样的 这个时候正是一年一度的毕业季 xff0c 而我最敬重的师兄即将要离校实习 xff0c 很幸运的是师兄收到了很不错的 offer xff0c 在这里
  • ubuntu系统下,下载安装Python程序的方法汇总(wget;apt-get;easy_install;pip)

    1 源码安装 xff1a 已知源码的地址 xff0c 例如 xff1a https www python org ftp python 3 6 1 Python 3 6 1 tgz 这是Python3 6 1的源码地址 xff0c 则可以使
  • 【C语言刷LeetCode】qsort库函数,刷题利器

    之前刷过一些leetcode算法题 xff0c 挺痛苦的 xff0c 毕竟用的C语言 其中很大一部分题都是考察数组和字符串 刷题中得到一个经验 xff0c 遇见数组先考虑排序 xff0c 排序就选qsort 那现在就总结写qsort的几个不
  • 一文讲解ARM、STM32之间的关系以及STM单片机介绍

    一 什么是ARM ARM xff1a xff08 Advanced RISC Machines xff09 高级精简指令集微处理器 它有几层含义 xff1a 1 ARM是一个公司 xff0c 英国公司 只出售芯片的技术授权 2 ARM是全球
  • ESP8266串口WiFi模块基本使用方法和配置教程

    前言 ESP8266是一款超低功耗的UART WiFi 透传模块 拥有业内极富竞争力的封装尺寸和超低能耗技术 专为移动设备和物联网应用设计 可将用户的物理设备连接到Wi Fi 无线网络上 进行互联网或局域网通信 实现联网功能 由于本人一直从
  • linux 内存查看方法:meminfo\maps\smaps\status 文件解析

    linux 下面查看内存有多种渠道 xff0c 比如通过命令 ps top free 等 xff0c 比如通过 proc系统 xff0c 一般需要比较详细和精确地知道整机内存 某个进程内存的使用情况 xff0c 最好通过 proc 系统 x
  • 数值型模板参数

    本篇文章学习记录 xff1a 数值型模板参数 实现C 43 43 数组类模板 1 模板中的数值型参数 模板参数可以是数值型参数 也就是非类型参数 如下图所示 xff1a 我们可以像上面定义一个局部数组 xff0c 但是却不能这样定义 xff
  • 矩阵分解 (加法篇)

    转自简书 xff1a https www jianshu com p fc89d92bbc24 引言 分解的思想其实并不古老 xff0c 而且大家都熟悉的 xff0c 把复杂的分而治之 xff0c 然后再组合起来 分解有什么好处 xff1f
  • 矩阵分解 (乘法篇)

    引自简书 xff1a https www jianshu com p 0741789ffd06 引言 前面我们在矩阵分解 加法篇 里面分析的加法下的矩阵分解 这里我们来看看乘法下矩阵分解的要点 对角和三角矩阵 首先 xff0c 我们总结下
  • openwrt 编译 问题

    xfeff xfeff 问题一 xff1a OpenWrt can only be built on a case sensitive filesystem 原因是文件解压保存在windows的文件夹 xff0c 应该要解压到linux环境
  • stm32 RTC_WaitForSynchro()死循环

    1 RTC WaitForSynchro 死循环 xff0c 发现是没有执行RTC Configuration 增加函数 xff0c 但不知道对之后的时钟准确性有什么影响 Function Name RTC Configuration De
  • 用docker创建ubuntu VNC桌面

    docker ubuntu vnc desktop from xff1a http wiki ros org docker Tutorials GUI 1 image 地址 https github com fcwu docker ubun
  • Win10如何安装VC6

    这里我们不真的运行setup安装VC6 xff0c 因为在win10安装程序有很大的概率会被卡住 xff0c 就算安装成功也是各种问题 xff0c 包括你设置了兼容性 这里有一个不用安装也能直接运行的办法 xff0c 并且可以支持C 43
  • 在 VMware 虚拟机中安装 open-vm-tools

    什么是 open vm tools xff1f open vm tools 是 VMware Tools 的开源实施 xff0c 由一套虚拟化实用程序组成 xff0c 这些程序可增强虚拟机在 VMware 环境中的功能 xff0c 使管理更
  • FreeRTOS任务管理与控制

    Task c文件 xff1a 全局变量 xff1a static xList pxReadyTasksLists configMAX PRIORITIES static xListxDelayedTaskList1 PRIVILEGED D
  • stm32 烧写下载失败 Error: Flash Download failed - "Cortex-M3"

    问题 xff1a Error Flash Download failed 34 Cortex M3 34 Load 34 10 10 axf 34 Set JLink Project File to 34 G vs keil test 10
  • 查看GIT项目在哪个路径下,查看GIT项目是从GIT的哪个分支上拉下来的命令

    转自 xff1a https www cnblogs com jishan coder p 8554782 html 1 根据路径找到本地存储地址 右键 如图 右键后可以看到有git Bash Here 前提是安装了git 点击 即可进入到

随机推荐

  • C++调用Python(混合编程)函数整理总结

    文章目录 C 43 43 调用python概述相关官方文档相关函数1 初始化python解释器环境2 调用python脚本的静态简单方式3 动态加载python模块并执行函数3 1不带参数和返回值的举例说明3 2带参数和返回值的举例说明 4
  • Markdown写作工具-Typora

    工具 Typora详解 零 文章目录 一 MarkDown 1 MarkDown是什么 Markdown 是一种轻量级标记语言 xff0c 它允许人们使用易读易写的纯文本格式编写文档 Markdown 语言在 2004 由约翰 格鲁伯 xf
  • git pull 强制覆盖本地的代码

    git pull 强制覆盖本地的代码方式 xff0c 下面是正确的方法 xff1a git fetch all 然后 xff0c 你有两个选择 xff1a git reset hard origin master 或者如果你在其他分支上 x
  • SMPL可视化大杀器,你并不需要下载SMPL就能可视化你的3D Pose

    SMPL 是一种3D人体建模方法 xff0c 现在几乎所有的元宇宙人体建模都是基于此类方法 xff0c 包括但不限于元宇宙 xff0c 自动驾驶等领域 它能估计出比较准确的人体3D姿态 xff0c 得益于海量数据训练的人体3D先验 不仅仅是
  • 程序员要多跳巢才能涨工资

    不要一辈子呆死在一家公司 都是打工高薪才是王道 fs xff1a 这 篇文章的本意 xff0c 是告诉大家如何识别公司 而不是鼓励大家无脑跳槽 只有当你在一个公司略有所成的时候 xff0c 你才能有所积累 跳槽更多时候 xff0c 应该看到
  • C++ 用libcurl库进行http通讯网络编程

    转自 http www cnblogs com moodlxs archive 2012 10 15 2724318 html 目录索引 xff1a 一 LibCurl基本编程框架 二 一些基本的函数 三 curl easy setopt函
  • 群晖NAS教程(一) 、利用Docker安装MySQL8并远程访问

    为了更好的浏览体验 欢迎光顾勤奋的凯尔森同学个人博客 群晖NAS教程 一 利用Docker安装MySQL8并远程访问 做为一个NAS发烧友玩家 在追求极致硬件配置的同时 也想在各个方面压榨一下自己的黑群晖 要不就对不起自己投入那么多的毛爷爷
  • 群晖NAS教程(二)、利用Docker安装Ubuntu并远程访问

    为了更好的浏览体验 欢迎光顾勤奋的凯尔森同学个人博客 群晖NAS教程 二 利用Docker安装Ubuntu并远程访问 作者 小景哥哥 一 下载镜像ubuntu upstart 一定要选这个镜像 双击运行 二 设置Ubuntu启动参数和端口号
  • 群晖NAS教程(五)、利用Docker安装Ubuntu-21.04并在Ubuntu上安装Redis进行远程访问

    为了更好的浏览体验 欢迎光顾勤奋的凯尔森同学个人博客 群晖NAS教程 五 利用Docker安装Ubuntu 21 04并在Ubuntu上安装Redis进行远程访问 由于上节我们安装的Ubuntu完全可以当做一个服务器来使用 这个可以完全替代
  • 群晖NAS教程(七)、利用Docker安装elasticsearch并进行远程访问

    为了更好的浏览体验 欢迎光顾勤奋的凯尔森同学个人博客 群晖NAS教程 七 利用Docker安装elasticsearch并进行远程访问 一 下载elasticsearch官方镜像 然后直接下载elasticsearch镜像即可 二 配置el
  • Airsim通过ros发布激光雷达数据+Lego-loam仿真测试(2)

    上篇博客只是简单跑通了流程 xff0c 存在的问题将在这篇进行修正 一 Lego loam里话题订阅 雷达点云话题为 xff1a velodyne points xff0c frame id为velodyne IMU话题为 xff1a Im
  • [控制算法]

    常用控制算法 0 博览众长 0 1 视频 1 DR CAN b站 0 2 文章 1 控制算法整理 0 3 传统 VS 现代控制算法 1 传统 传统控制算法 xff1a PID xff0c 模糊 xff0c 神经网络控制算法 2 现代 现代控
  • 七、输入/输出流--streambuffer类介绍--

    缓冲区类 类模板定义为basic streambuf xff0c 由 lt iostream gt 给出 xff1a 1 stream缓冲区 通常stream不负责实际读写操作 xff0c 而是stream buffer实现streambu
  • 七、输入/输出流--streambuffer类介绍--自定义缓冲区

    基本上没看懂 xff0c 那个大神如果可以的话 xff0c 推荐一点相关资料 xff0c 真的不太明白这个缓冲区的内部原理 3 自定义缓冲区 缓冲区有basic streambuf定义 xff0c 针对字型为char和wchar标准库提供了
  • 串口通信学习(GPS模块)2021.5.10

    GPS串口通信学习实践 2021 5 10 1 串口通信简介1 1 波特率1 2 数据位1 3 停止位1 4 奇偶校验位 2 GPS模块串口通信配置2 1 驱动安装2 2 插入GPS模块2 3 GPS模块串口通信数据简介 3 Java实现G
  • Lambda表达式的使用

    对于jdk1 8其实并不是那么熟悉 xff0c 但是要学习这一点 xff0c 对以后工作有好处 xff0c 接下来开始学习jdk1 8在Android studio的配置以及lambda表达式的使用吧 Lambda表达式 jdk1 8中新增
  • ROS实现无人机目标跟踪/物体跟随/循迹

    无人机自主物体跟随 循迹 1 物体跟踪1 1 实现思路1 2 代码示例 2 自主寻线 本实验采用ROS和OpenCV实现功能 xff0c 实验平台采用Parrot的Bebop2无人机ROS部分的学习可以参考我的专栏 xff1a ROS学习记
  • vs code中项目的基本配置--include路径、运行参数、debug配置

    1 安装C C 43 43 for Visual Studio Code 点击左边扩展栏图标 gt 搜索C C 43 43 gt 安装 gt Reload xff1a 安装完成之后 xff0c 打开你的包含c 43 43 的文件夹 xff0
  • CMakeLists.txt编写规范模板及CMake常用变量

    文章目录 基本语法规则常见CMakeLists txt中指令剖析从VS项目配置过程理解CMakeLists内容CMake中常用变量汇总常用CMakeLists文件模板 基础模板使用OpenCV库CMakeLists文件模板使用PCL库CMa
  • C++ 多线程detach()操作的坑以及传参

    detach 的作用是将子线程和主线程的关联分离 xff0c 也就是说detach 后子线程在后台独立继续运行 xff0c 主线程无法再取得子线程的控制权 xff0c 即使主线程结束 xff0c 子线程未执行也不会结束 当主线程结束时 xf