C++:线程(thread)的创建、调用及销毁

news2024/11/13 10:03:02

在 C++ 中,线程的管理主要依赖于标准库 std::thread,自 C++11 起,这一功能被标准化,使得我们能够更加方便地创建、管理和销毁线程。这里我们详细讲解线程的创建、调用和销毁流程。

1. 线程的创建

创建线程通常是为了在单独的线程中执行某个任务。我们可以通过 std::thread 对象来创建一个新的线程。一个线程可以从以下几种类型的可调用对象启动:

  • 普通函数
  • Lambda 表达式
  • 函数对象
  • 类的成员函数

1.1 使用普通函数

#include <iostream>
#include <thread>

void printMessage() {
    std::cout << "Hello from thread!" << std::endl;
}

int main() {
    std::thread myThread(printMessage); // 创建一个线程并启动 printMessage
    myThread.join();                    // 等待线程完成
    return 0;
}
  • std::thread myThread(printMessage) 创建一个线程对象 myThread,并启动执行 printMessage 函数。
  • myThread.join() 等待线程完成。如果没有调用 join()detach(),程序在结束时会崩溃。

1.2 使用 Lambda 表达式

#include <iostream>
#include <thread>

int main() {
    std::thread myThread([]() {
        std::cout << "Hello from lambda!" << std::endl;
    });
    myThread.join();
    return 0;
}
  • 这里我们创建了一个线程并使用 lambda 表达式作为线程函数。Lambda 允许我们在局部作用域中定义线程任务。

1.3 使用函数对象

#include <iostream>
#include <thread>

class PrintTask {
public:
    void operator()() const {
        std::cout << "Hello from function object!" << std::endl;
    }
};

int main() {
    std::thread myThread(PrintTask()); // 创建线程并启动
    myThread.join();
    return 0;
}
  • 这是通过重载 operator() 来定义一个可调用的对象,该对象可以直接用来创建线程。

1.4 使用类的成员函数

#include <iostream>
#include <thread>

class MyClass {
public:
    void memberFunction() {
        std::cout << "Hello from member function!" << std::endl;
    }
};

int main() {
    MyClass obj;
    std::thread myThread(&MyClass::memberFunction, &obj); // 需要传递对象指针
    myThread.join();
    return 0;
}
  • 如果是成员函数,则需要传递对象指针。&MyClass::memberFunction 表示成员函数地址,&obj 是指向 MyClass 实例的指针。

2. 线程的调用

  • 线程的参数传递:你可以向线程函数传递参数,它们会按照值传递的方式进行复制。为了传递引用,可以使用 std::refstd::cref

#include <iostream>
#include <thread>

void printNumber(int n) {
    std::cout << "Number: " << n << std::endl;
}

int main() {
    int value = 42;
    std::thread myThread(printNumber, value);
    myThread.join(); // 等待线程完成
    return 0;
}
  • 捕获引用
void increment(int& n) {
    ++n;
}

int main() {
    int value = 0;
    std::thread myThread(increment, std::ref(value));
    myThread.join();
    std::cout << "Value after increment: " << value << std::endl;
    return 0;
}

这里 std::ref 确保 value 以引用的形式传递。

3. 线程的同步

  • join():主线程会等待 myThread 结束。join 是同步机制,用于确保线程完成后主线程才会继续。
std::thread myThread(task);
myThread.join();
  • detach():将线程从主线程分离,让它独立运行。独立运行的线程在后台执行,主线程不再等待它完成。需谨慎使用,可能引发访问冲突。
std::thread myThread(task);
myThread.detach();

4. 线程的销毁

  • std::thread 对象离开作用域时,如果没有调用 join()detach(),程序会触发异常终止。
  • 使用 join() 可以让主线程等待子线程完成,从而安全地销毁线程。
  • 使用 detach() 可以将线程从 std::thread 对象中分离,使其成为独立线程。调用 detach() 后,std::thread 对象不再管理该线程。

5. 线程的生命周期管理

  • RAII:考虑使用 RAII 类管理线程生命周期,确保在对象析构时 join()detach() 线程,从而避免泄漏和不正确的管理。
class ThreadGuard {
public:
    explicit ThreadGuard(std::thread& t) : thread(t) {}
    ~ThreadGuard() {
        if (thread.joinable()) {
            thread.join();
        }
    }

private:
    std::thread& thread;
};

6. 线程的注意事项

  • 避免数据竞争和同步问题:线程共享数据时要小心。可以使用 std::mutex 进行保护,或者使用其他同步机制如 std::lock_guard
  • 避免内存泄漏:如果使用 new thread(),确保 delete 以释放分配的内存。
  • 检查 joinable() 状态:调用 join()detach() 前,可以用 joinable() 检查线程状态。

 7. 示例总结

#include <iostream>
#include <thread>

void task(int id) {
    std::cout << "Thread " << id << " is running." << std::endl;
}

int main() {
    std::thread t1(task, 1); // 创建线程并传递参数
    std::thread t2(task, 2);

    t1.join(); // 等待 t1 完成
    t2.join(); // 等待 t2 完成

    return 0;
}

8. 线程管理总结

  • 使用 std::thread 创建和管理线程。
  • join()detach() 用于控制线程的生命周期。
  • 避免重复 join()detach(),确保资源管理得当。
  • 使用同步机制保护共享数据的访问。

通过这种方式,你可以更灵活地创建、管理和销毁 C++ 线程,确保程序的并发性和资源管理的安全性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2237100.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

详解如何创建SpringBoot项目

目录 点击New Project 选择依赖 简单使用SpringBoot 前面已经讲解了如何获取IDEA专业版&#xff0c;下面将以此为基础来讲解如何创建SpringBoot项目。 点击New Project 选择依赖 注意&#xff0c;在选择SpringBoot版本时&#xff0c;不要选择带SNAPSHOT的版本。 这样&#…

点云分割总结

点云分割总结 point transformerbackground 标量自注意力和向量自注意力&#xff08;可参考论文&#xff09;标量自注意力向量注意力 Point Transformer Layer下采样上采样整体结构 point transformer v2group vector attentionPosition Encoding MultiplerPartition-based Poo…

智象未来(HiDream.ai):从科技创新启程,绘制智能未来新篇章

在人工智能领域飞速演进的当下&#xff0c;智象未来&#xff08;HiDream.ai&#xff09;作为全球领先的多模态生成式人工智能技术供应商&#xff0c;正以其独树一帜的视觉多模态大模型及创新应用&#xff0c;推动行业趋势的前进。智象未来&#xff08;HiDream.ai&#xff09;自…

CSP/信奥赛C++刷题训练:经典例题 - 栈(2):洛谷P1981 :[NOIP2013 普及组] 表达式求值

CSP/信奥赛C刷题训练&#xff1a;经典例题 - 栈&#xff08;2&#xff09;&#xff1a;洛谷P1981 &#xff1a;[NOIP2013 普及组] 表达式求值 题目背景 NOIP2013 普及组 T2 题目描述 给定一个只包含加法和乘法的算术表达式&#xff0c;请你编程计算表达式的值。 输入格式 …

LVGL加入外围字库

一、首先lvgl是有自带字库的 lvgl/src/font 如下图 二、但如果这个字库不能满足我们的需求我们就要外建字库。 1、字库生成软件LVGL官网&#xff0c;字体转换器 — LVGL如下图&#xff1a; 最后按“提交”就可以看到有一个字体被下载到你电脑里。他是以.c文件的型式&#xff0…

创新引领,模块化微电网重塑能源格局

根据QYResearch调研团队最新发布的《全球模块化微电网市场报告2023-2029》显示&#xff0c;预计到2029年&#xff0c;全球模块化微电网市场的规模将扩大至33.1亿美元&#xff0c;且在未来几年内&#xff0c;其年复合增长率&#xff08;CAGR&#xff09;将达到8.8%。 如下图所示…

FPGA 第4讲 初识Verilog HDL

时间&#xff1a;2024.11.9 一、学习内容 1.Verilog HDL简介 1.1语言简介 Verilog HDL是一种硬件描述语言&#xff0c;以文本的形式来描述数字系统硬件的结构和行为的语言&#xff0c;用它可以表示逻辑电路图、逻辑表达式&#xff0c;还可以表示数字逻辑系统所完成的逻辑功能…

【51单片机】LED点阵屏 原理 + 使用

学习使用的开发板&#xff1a;STC89C52RC/LE52RC 编程软件&#xff1a;Keil5 烧录软件&#xff1a;stc-isp 开发板实图&#xff1a; 文章目录 LED点阵屏显示原理74HC595 编码LED点阵屏显示笑脸LED点阵屏显示动画 LED点阵屏 点阵屏在开发板的右上角&#xff0c;注意使用前需要…

Chrome扩展是程序员做独立开发的绝佳入场机会

一、开发成本低&#xff0c;难度低 简便灵活&#xff1a;相比开发移动应用&#xff0c;浏览器扩展的开发过程更加简便灵活&#xff0c;更适合初学者。省时省力&#xff1a;通过扩展&#xff0c;你可以修改现有网站的功能&#xff0c;无需从零开始搭建应用&#xff0c;大大节省…

记录一下最近遇到的两个问题

问题1 网友问&#xff1a;一个数据同步的程序之前运行正常&#xff0c;突然数据有问题了&#xff0c;俺的回答是退出杀毒软件 问题是很快解决了&#xff0c;但是网友后来说&#xff0c;客户觉得程序很不稳定。俺不清楚这算不算背锅。 问题2 今天下午&#xff0c;调试着程序蓝…

30.1 时序数据库TSDB的典型特点

本节重点介绍 : db-ranking网站对db进行排名时序数据特点时序数据库特点时序数据库遇到的挑战开源时间序列数据库 db-ranking 一个神奇的网站 https://db-engines.com/en/ranking 时序数据ranking https://db-engines.com/en/ranking/timeseriesdbms 排名方法 https://db-en…

Linux SSH私钥认证结合cpolar内网穿透安全高效远程登录指南

文章目录 前言1. Linux 生成SSH秘钥对2. 修改SSH服务配置文件3. 客户端秘钥文件设置4. 本地SSH私钥连接测试5. Linux安装Cpolar工具6. 配置SSHTCP公网地址7. 远程SSH私钥连接测试8. 固定SSH公网地址9. 固定SSH地址测试 前言 开发人员在工作中经常需要远程访问服务器和数据中心…

vscode摸鱼学习插件开发

不知道大家在摸鱼的时候&#xff0c;会不会想要学习&#xff1f; 或者有没有考公人&#xff0c;下班要学习的&#xff1f; 上班时间摸鱼&#xff0c;下班时间不够学习&#xff1f; 为此&#xff0c;我决定开发一个vscode插件&#xff0c;来刷粉笔题 粉笔插件名称&#xff1a;…

深入浅出WebSocket(实践聊天室demo)

文章目录 什么是WebSocket?WebSocket连接过程WebSocket与Http的区别重连机制完整代码使用方法心跳机制实现聊天室demo(基于Socket.io)参考文章、视频小广告~什么是WebSocket? WebSocket 是一种在单个TCP连接上进行全双工通信的协议(计算机网络应用层的协议) 在 WebSocket A…

[ Linux 命令基础 7 ] Linux 命令详解-磁盘管理相关命令

&#x1f36c; 博主介绍 &#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 _PowerShell &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【数据通信】 【通讯安全】 【web安全】【面试分析】 &#x1f389;点赞➕评论➕收藏 养成习…

ElasticSearch 添加IK分词器

ElasticSearch 添加IK分词器 前言一、IK分词器的算法二、Ik分词器的下载安装&#xff08;Winows 版本&#xff09;三、Ik分词器的下载安装&#xff08;Linux 版本&#xff09;四、验证测试&#xff08;postman工具&#xff09;测试 ik_smart 分词算法测试 ik_max_word 分词算法…

aws(学习笔记第十一课) 使用AWS的EFS,以及AWS Storage Gateway

aws(学习笔记第十一课) 使用AWS的EFS和AWSStorage Gateway 学习内容&#xff1a; 使用AWS的EFS使用AWS Storage Gateway 1. 使用AWS的EFS 什么是EFS EFS是 Elastic File System的缩写。前面练习的实例存储和EBS都是同时只能一个EC2实例进行挂载&#xff0c;不能实现多个EC2实…

Diffusion Policy——斯坦福刷盘机器人UMI所用的扩散策略(含Diff-Control、ControlNet详解)

前言 本文一开始是属于此文《UMI——斯坦福刷盘机器人&#xff1a;从手持夹持器到动作预测Diffusion Policy(含代码解读)》的第三部分&#xff0c;考虑后Diffusion Policy的重要性很高&#xff0c;加之后续还有一系列基于其的改进工作 故独立成本文&#xff0c;且把原属于另一…

计算机毕业设计 | SpringBoot慈善公益平台 爱心互助活动发布管理系统(附源码)

1&#xff0c;项目介绍 爱慈善公益平台&#xff08;love-charity&#xff09;是一个基于 SpringBoot 开发的标准 Java Web 项目。整体页面非常的简约大气&#xff0c;项目的完整度较高&#xff0c;是一个偏向公益论坛的系统。非常适合刚刚接触学习 SpringBoot 的技术小白学习&…

【深入浅出】之Linux进程(二)

&#x1f4c3;博客主页&#xff1a; 小镇敲码人 &#x1f49a;代码仓库&#xff0c;欢迎访问 &#x1f680; 欢迎关注&#xff1a;&#x1f44d;点赞 &#x1f442;&#x1f3fd;留言 &#x1f60d;收藏 &#x1f30f; 任尔江湖满血骨&#xff0c;我自踏雪寻梅香。 万千浮云遮碧…