原始C++标准仅支持单线程编程。新的C++标准(称为c++11或c++0x)于2011年发布。在c++11中,引入了新的线程库。因此运行本文程序需要C++至少符合c++11标准。

3 小心地将参数传递给线程

要将参数传递给线程的关联可调用对象或函数,只需将其他参数传递给std::thread构造函数。默认情况下,所有参数都复制到新线程的内部存储中。让我们看一个例子:

3.1 在c++11中将简单参数传递给std::thread

  1. #include <iostream>
  2. #include <string>
  3. #include <thread>
  4. void threadCallback(int x, std::string str)
  5. {
  6. std::cout << "Passed Number = " << x << std::endl;
  7. std::cout << "Passed String = " << str << std::endl;
  8. }
  9. int main()
  10. {
  11. int x = 10;
  12. std::string str = "Sample String";
  13. std::thread threadObj(threadCallback, x, str);
  14. threadObj.join();
  15. return 0;
  16. }

输出为:

  1. Passed Number = 10
  2. Passed String = Sample String

3.2 如何不将参数传递给c++11中的线程

不要将变量的地址从本地堆栈传递到线程的回调函数。因为线程1中的局部变量可能超出了作用域,但是线程2仍然试图通过它的地址访问它。在这种情况下,访问无效地址可能会导致意外行为。例如:

  1. #include <iostream>
  2. #include <thread>
  3. void newThreadCallback(int * p)
  4. {
  5. std::cout << "Inside Thread:"":p = " << p << std::endl;
  6. std::chrono::milliseconds dura(1000);
  7. std::this_thread::sleep_for(dura);
  8. *p = 19;
  9. }
  10. void startNewThread()
  11. {
  12. int i = 10;
  13. std::cout << "Inside Main Thread:"":i = " << i << std::endl;
  14. std::thread t(newThreadCallback, &i);
  15. t.detach();
  16. std::cout << "Inside Main Thread:"":i = " << i << std::endl;
  17. }
  18. int main()
  19. {
  20. startNewThread();
  21. // 表示一段时间,这里是2000 毫秒
  22. std::chrono::milliseconds dura(2000);
  23. // 当前线程休眠一段时间
  24. std::this_thread::sleep_for(dura);
  25. return 0;
  26. }

输出为:

  1. Inside Main Thread::i = 10
  2. Inside Thread::p = Inside Main Thread::i = 10
  3. 000000D9DD5BF4A4
  4. 程序崩溃

同样,将指向堆上内存的指针传递给线程时也要小心。因为在新线程试图访问该存储器之前,某些线程可能会删除该存储器。在这种情况下,访问无效地址可能导致意外行为。例如:

  1. #include <iostream>
  2. #include <thread>
  3. void newThreadCallback(int * p)
  4. {
  5. std::cout << "Inside Thread : "" : p = " << p << std::endl;
  6. std::chrono::milliseconds dura(1000);
  7. std::this_thread::sleep_for(dura);
  8. *p = 19;
  9. }
  10. void startNewThread()
  11. {
  12. int * p = new int();
  13. *p = 10;
  14. std::cout << "Inside Main Thread : "" : *p = " << *p << std::endl;
  15. std::thread t(newThreadCallback, p);
  16. t.detach();
  17. delete p;
  18. p = NULL;
  19. }
  20. int main()
  21. {
  22. startNewThread();
  23. std::chrono::milliseconds dura(2000);
  24. std::this_thread::sleep_for(dura);
  25. return 0;
  26. }

输出为:

  1. Inside Main Thread :? : *p = 10
  2. Inside Thread :? : p = 0000024AC61ECEA0

3.3 如何在c++11中传递对std::thread的引用

由于参数被复制到新的线程堆栈,因此,如果您需要以通用方式传递引用,即

  1. #include <iostream>
  2. #include <thread>
  3. void threadCallback(int const & x)
  4. {
  5. int & y = const_cast<int &>(x);
  6. y++;
  7. std::cout << "Inside Thread x = " << x << std::endl;
  8. }
  9. int main()
  10. {
  11. int x = 9;
  12. std::cout << "In Main Thread:Before Thread Start x = " << x << std::endl;
  13. std::thread threadObj(threadCallback, x);
  14. threadObj.join();
  15. std::cout << "In Main Thread:After Thread Joins x = " << x << std::endl;
  16. return 0;
  17. }

输出为:

  1. In Main Thread:Before Thread Start x = 9
  2. Inside Thread x = 10
  3. In Main Thread:After Thread Joins x = 9

即使threadCallback接受参数作为参考,但所做的更改仍在线程外部不可见。这是因为线程函数threadCallback中的x引用了在新线程的堆栈上复制的临时值。如何解决呢?使用std::ref()即可。std::ref 用于包装按引用传递的值。

  1. #include <iostream>
  2. #include <thread>
  3. void threadCallback(int const & x)
  4. {
  5. int & y = const_cast<int &>(x);
  6. y++;
  7. std::cout << "Inside Thread x = " << x << std::endl;
  8. }
  9. int main()
  10. {
  11. int x = 9;
  12. std::cout << "In Main Thread : Before Thread Start x = " << x << std::endl;
  13. std::thread threadObj(threadCallback, std::ref(x));
  14. threadObj.join();
  15. std::cout << "In Main Thread : After Thread Joins x = " << x << std::endl;
  16. return 0;
  17. }

输出为:

  1. In Main Thread : Before Thread Start x = 9
  2. Inside Thread x = 10
  3. In Main Thread : After Thread Joins x = 10

3.4 将指向类成员函数的指针分配为线程函数

将指针传递给成员函数作为回调函数,并将指针传递给Object作为第二个参数。例如:

  1. #include <iostream>
  2. #include <thread>
  3. class DummyClass {
  4. public:
  5. DummyClass()
  6. {}
  7. DummyClass(const DummyClass & obj)
  8. {}
  9. void sampleMemberFunction(int x)
  10. {
  11. std::cout<<"Inside sampleMemberFunction "<<x<<std::endl;
  12. }
  13. };
  14. int main() {
  15. DummyClass dummyObj;
  16. int x = 10;
  17. std::thread threadObj(&DummyClass::sampleMemberFunction,&dummyObj, x);
  18. threadObj.join();
  19. return 0;
  20. }

输出为:

  1. Inside sampleMemberFunction 10

3.5 参考

https://thispointer.com//c11-multithreading-part-4-data-sharing-and-race-conditions/

[编程基础] C++多线程入门3-小心地将参数传递给线程的更多相关文章

  1. [编程基础] C++多线程入门6-事件处理的需求

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++11标准. 文章目录 6 事件处 ...

  2. [编程基础] C++多线程入门4-数据共享和资源竞争

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++ 11标准. 4 数据共享和资源 ...

  3. [编程基础] C++多线程入门7-条件变量介绍

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++11标准. 文章目录 7 条件变 ...

  4. [编程基础] C++多线程入门5-使用互斥锁解决资源竞争

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++11标准. 文章目录 5 使用互 ...

  5. [编程基础] C++多线程入门2-连接和分离线程

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++11标准. 文章目录 2 连接和 ...

  6. [编程基础] C++多线程入门8-从线程返回值

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++11标准. 8 从线程返回值 8 ...

  7. [编程基础] C++多线程入门1-创建线程的三种不同方式

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++11标准. 1 创建线程的三种不 ...

  8. [编程基础] C++多线程入门10-packaged_task示例

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++11标准. 文章目录 10 pa ...

  9. [编程基础] C++多线程入门9-async教程和示例

    原始C++标准仅支持单线程编程.新的C++标准(称为C++11或C++0x)于2011年发布.在C++11中,引入了新的线程库.因此运行本文程序需要C++至少符合C++11标准. 文章目录 9 asy ...

随机推荐

  1. SpringBoot(一) - SpringBoot 初识

    1.创建SpringBoot项目 1.1 使用Spring Initializr 的 Web页面创建项目 创建网址:https://start.spring.io/ 1.2 使用IDEA创建 省略: ...

  2. List接口中的常用方法

    void add(int index, Object ele):在index位置插入ele元素boolean addAll(int index, Collection eles):从index位置开始 ...

  3. git clone开启云上AI开发

    摘要:相比于传统的软件开发,AI开发存在以下4个痛点:算法繁多:训练时间长:算力需求大:模型需手动管理,我们可以使用云上AI开发的方式来缓解以上4个痛点. 本文分享自华为云社区<git clon ...

  4. C语言爱心表白程序

    #include <stdio.h> #include <math.h> #include <windows.h> #include <tchar.h> ...

  5. LcdToos设置“自动播放”和“上电自动开机”的作用

    "自动播放"功能,常用于屏演示或者老化功能,使能后,按开关点亮屏,PX01会自动按"画面定制"栏中进行自动顺序播放:开启方法如下: 打开相应的点屏工程,在&qu ...

  6. letcode刷题记录-day03-罗马转整数

    题目 罗马转整数 题目描述 罗马数字包含以下七种字符: I, V, X, L,C,D 和 M. 字符 数值 I 1 V 5 X 10 L 50 C 100 D 500 M 1000 例如, 罗马数字 ...

  7. SpringCloud(六) - RabbitMQ安装,三种消息发送模式,消息发送确认,消息消费确认(自动,手动)

    1.安装erlang语言环境 1.1 创建 erlang安装目录 mkdir erlang 1.2 上传解压压缩包 上传到: /root/ 解压缩# tar -zxvf otp_src_22.0.ta ...

  8. JAVA开发搞了一年多的大数据,究竟干了点啥

    JAVA开发搞了一年多大数据的总结 ​ 2021年7月份加入了当前项目组,以一个原汁原味的Java开发工程师的身份进来的,来了没多久,项目组唯一一名大数据开发工程师要离职了,一时间一大堆的数据需求急需 ...

  9. 真正“搞”懂HTTP协议03之时间穿梭

    上一篇我们简单的介绍了一下DoD模型和OSI模型,还着重的讲解了TCP的三次握手和四次挥手,让我们在空间层面,稍稍宏观的了解了HTTP所依赖的底层模型,那么这一篇,我们来追溯一下HTTP的历史,看一看 ...

  10. bugku web基础$_POST

    这道题也是让what=flag就行了 直接试试通过max hackbar来进行post传入 得到flag