thread---基本使用和常见错误

news/2025/2/22 2:12:13

多线程基础

一、C++多线程基础

  1. 线程概念:线程是操作系统能够调度的最小执行单元,同一进程内的多个线程共享内存空间
  2. 头文件#include <thread>
  3. 线程生命周期:创建->执行->销毁(需显式管理)
  4. 注意事项
    • 线程函数参数建议使用值传递
    • 注意数据竞争问题(需配合互斥锁使用)
    • 主线程退出前必须处理所有子线程

二、核心函数详解

1. std::thread()
// 创建线程的三种方式
void func1();
void func2(int num);

// 方式1:普通函数
std::thread t1(func1);

// 方式2:带参数的函数
std::thread t2(func2, 42);

// 方式3:Lambda表达式
std::thread t3([](){
    std::cout << "Lambda thread" << std::endl;
});

特性

  • 构造函数立即启动线程
  • 参数自动转发给线程函数
  • 必须处理线程对象(join或detach)
2. join()
std::thread t(func);
t.join(); // 阻塞当前线程,直到t执行完成

特点

  • 同步线程执行顺序
  • 每个线程对象只能调用一次
  • 调用后线程对象不再关联实际线程
3. joinable()
if(t.joinable()) {
    t.join(); // 或t.detach()
}

判断条件

  • 线程已被创建(关联实际系统线程),正在进行的线程返回true
  • 尚未被join或detach
4. detach()
std::thread t(func);
t.detach(); // 分离线程

特点

  • 线程在后台独立运行(守护线程)
  • 失去对线程的直接控制
  • 必须确保线程函数内的资源有效性

三、使用示例

#include <iostream>
#include <thread>

void print_num(int num) {
    std::cout << "Number: " << num << std::endl;
}

int main() {
    std::thread worker(print_num, 42);
    
    if(worker.joinable()) {
        worker.join(); // 或detach()
    }
    
    return 0;
}

线程函数中的数据未定义的错误

1.临时变量导致的传参错误

传递临时变量,临时变量在thread函数结束后销毁,没有传递到func里,导致未定义错误

#include<iostream>
#include<thread>
void func(int& x) {
	x += 1;
}
int main() {
	std::thread t(func, 1); //传递临时变量,临时变量在thread函数结束后销毁,没有传递到func里,导致未定义错误
	t.join();
	return 0;
}

解决方法:创建一个持久变量

#include<iostream>
#include<thread>
void func(int& x) {
	x += 1;
	std::cout << x << " ";
}
int main() {
	int x = 1; //将变量复制到一个持久的对象
	std::thread t(func, std::ref(x));//将变量的引用传递给线程
	t.join();
	return 0;
}

2.传递指针或引用指向局部变量的问题

test函数执行结束,a立刻释放,无法继续传进线程中的func函数中

#include <iostream>
#include<thread>

std::thread t;
void func(int& x) {
	x += 1;
}
void test() {
	int a = 1;
	t = std::thread(func, std::ref(a));
}
int main()
{
	test();
	t.join();
	return 0;
}

解决办法,把a 放到外面,a 变成一直可以被取到的全局变量

#include <iostream>
#include<thread>

std::thread t;
int a = 1;

void func(int& x) {
	x += 1;
}

void test() {
	
	t = std::thread(func, std::ref(a));
}
int main()
{
	test();
	t.join();
	return 0;
}

3. 传递指针或引用指向已释放的内存的问题

指针在传进去之前delete,导致传进去的是空指针(不指向1)

#include<iostream>
#include<thread>

std::thread t;
void func(int* x) {
	std::cout << *x << std::endl;
}

int main() {
	int* ptr = new int(1);
	std::thread t(func, ptr);
	delete ptr;

	t.join();
	return 0;
}

解决办法
取消提前释放即可或使用智能指针

4. 类成员函数作为入口函数,类对象被提前释放

#include<iostream>
#include<thread>
#include<Windows.h>

class A {
public:
	void func() {
		Sleep(1000);
		std::cout << "6" << std::endl;
	}

};

int main() {
	A *a;
	std::thread t(&A::func, a);
	delete a;
	
	t.join();
}

解决办法:使用智能指针

#include<iostream>
#include<thread>
#include<Windows.h>

class A {
public:
	void func() {
		Sleep(1000);
		std::cout << "6" << std::endl;
	}
};

int main() {
	std::shared_ptr<A> a = std::make_shared<A>();//使用智能指针地址一直有效,不会出现提前释放的问题
	std::thread t(&A::func, a);
	
	t.join();
}

5.入口函数为类的私有成员函数

#include<iostream>
#include<thread>
#include<Windows.h>

class A {
private:
	
	void func() {
		Sleep(1000);
		std::cout << "6" << std::endl;
	}
};

void thread_func() {
	std::shared_ptr<A> a = std::make_shared<A>();
	std::thread t(&A::func, a);
	t.join();
}

int main() {
	
	thread_func();
}

解决办法:使用友元函数

#include<iostream>
#include<thread>
#include<Windows.h>

class A {
private:
	friend void thread_func();
	void func() {
		Sleep(1000);
		std::cout << "6" << std::endl;
	}
};

void thread_func() {
	std::shared_ptr<A> a = std::make_shared<A>();
	std::thread t(&A::func, a);
	t.join();
}

int main() {
	
	thread_func();
}

朝饮花上露,夜卧松下风。
云英化为水,光采与我同。 —王昌龄


http://www.niftyadmin.cn/n/5861520.html

相关文章

如何将MySQL数据库迁移至阿里云

将 MySQL 数据库迁移至阿里云可以通过几种不同的方法&#xff0c;具体选择哪种方式取决于你的数据库大小、数据复杂性以及对迁移速度的需求。阿里云提供了多种迁移工具和服务&#xff0c;本文将为你介绍几种常见的方法。 方法一&#xff1a;使用 阿里云数据库迁移服务 (DTS) 阿…

C语言基础系列【15】union 共用体

博主介绍&#xff1a;程序喵大人 35- 资深C/C/Rust/Android/iOS客户端开发10年大厂工作经验嵌入式/人工智能/自动驾驶/音视频/游戏开发入门级选手《C20高级编程》《C23高级编程》等多本书籍著译者更多原创精品文章&#xff0c;首发gzh&#xff0c;见文末&#x1f447;&#x1f…

前端PDF转图片技术调研实战指南:从踩坑到高可用方案的深度解析

本文以真实业务场景为背景&#xff0c;深入剖析前端PDF转图片的 7大核心指标 &#xff0c;通过3000字详解5种方案对比性能压测数据&#xff0c;输出可复用的技术调研方法论。 一、技术调研认知误区与破局之道 1.1 需求理解典型翻车现场 // 错误案例&#xff1a;未明确需求边界…

【系统架构设计师】需求工程

目录 1. 说明2. 软件需求3. 需求阶段4. 需求管理5. 例题5.1 例题1 1. 说明 1.软件需求目前并没有统一的定义&#xff0c;但都包含以下几方面的内容。2.用户解决问题或达到目标所需条件或权能&#xff08;Capability&#xff09;。系统或系统部件要满足合同、标准、规范或其他正…

2月17日c语言框架

C语言框架以及常用函数 1&#xff0c;scanf&#xff08;&#xff09; 2&#xff0c;printf&#xff08;&#xff09; 1scanf&#xff08;&#xff09;函数的特点&#xff0c;必须填写参数&#xff0c;不然一定会报错 2输入一个整数&#xff0c;按回车之后&#xff0c;会出…

alt+tab切换导致linux桌面卡死的急救方案

环境 debian12 gnome43.9 解决办法 观察状态栏&#xff0c;其实系统是没有完全死机的&#xff0c;而且gnome也可能没有完全死机。 1. alt f4 关闭桌面上的程序&#xff0c;因为这个方案是我刚刚看到的&#xff0c;所以不确定能不能用&#xff0c;比起重启系统&#xff0c;…

嵌入式 Linux:使用设备树驱动GPIO全流程

文章目录 前言 一、设备树配置 1.1 添加 pinctrl 节点 1.2 添加 LED 设备节点 二、编写驱动程序 2.1 驱动程序框架 2.2 编译驱动程序 三、测试 总结 前言 在嵌入式 Linux 开发中&#xff0c;设备树&#xff08;Device Tree&#xff09;和 GPIO 子系统是控制硬件设备的重要工具…

多人协同开发 —— Git Aoneflow工作流

一、Aoneflow工作流核心架构 #mermaid-svg-rwTOe9qYwzG3wkdy {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-rwTOe9qYwzG3wkdy .error-icon{fill:#552222;}#mermaid-svg-rwTOe9qYwzG3wkdy .error-text{fill:#552222…