《C++避坑神器·十九》C++多线程使用,啥也不懂看它就对了

简介: 《C++避坑神器·十九》C++多线程使用,啥也不懂看它就对了

C11后有了标准的线程库:

#include <thread>

并发

是指多个线程任务在同一个CPU上快速地轮换执行,由于切换的速度非常快,给人的感觉就是这些线程任务是在同时进行的,但其实并发只是一种逻辑上的同时进行;

并行

是指多个线程任务在不同CPU上同时进行,是真正意义上的同时执行

多进程是指操作系统能同时运行多个任务(程序)。

多线程是指在同一程序中有多个顺序流在执行。

进程是资源分配的最小单位,线程是CPU调度的最小单位,一个程序至少有一个进程,一个进程至少有一个线程。

注意:如果想要两个不同的线程调用同一函数,可以thread th1(fun1),thread th2(fun1),这样相当于同一时间有两个fun1方法在同步执行。

带参数写法:

void fun3(int i)
{
  //  多线程任务
}
 thread th3(fun3, 10);  //带参数

1、最后为啥要加join()?这里创建了两个线程,还有个主线程,三个线程各自走各自的互不干扰,当主线程走完时,这两个线程可能还没走完,即线程仍然存在,又由于线程的对象是在主线程创建的,主线程走完,线程对象便销毁,但子线程还存在这就导致崩溃。所以最后要加join让主线程先等子线程走完在结束。(调用了join子线程不在执行,join只能调用一次),join会阻塞当前线程,所以主线程先走到join这就被阻塞,等待子线程执行完join才会继续走。注意在调用join之前记得给子线程一个出口让死循环结束。

2、上述问题还可以用detach来解决,detach是用来和线程对象分离的,这样线程可以独立地执行,不过这样由于没有thread对象指向该线程而失去了对它的控制,当对象析构时线程会继续在后台执行,但是当主程序退出时并不能保证线程能执行完。如果没有良好的控制机制或者这种后台线程比较重要,最好不用detach而应该使用join。

简单说join会阻塞当前线程,detach不会阻塞当前线程

有个注意点:线程可以有返回值,上面操作可以这样做

thread th1;  
th1 = thread(fun1);

有些线程没有被创建或者已经被join/detach,就不能再次被join/detach,可以用joinable()判断当前线程是否可以被join/detach

线程同步:

头文件是,mutex是用来保证线程同步的,防止不同的线程同时操作同一个共享数据。

但是使用mutex是不安全的,当一个线程在解锁之前异常退出了,那么其它被阻塞的线程就无法继续下去。

使用lock_guard则相对安全,它是基于作用域的,能够自解锁,当该对象创建时,它会像m.lock()一样获得互斥锁,当生命周期结束时,它会自动析构(unlock),不会因为某个线程异常退出而影响其他线程。

lock_guard作用域取决于其所在区域,如果是栈内存,也其作用域是其最近的一对花括号,如果是堆内存,其生命周期取决于手动释放的时机,总之,就是和普通对象的生命周期相同。构造的时候加锁,析构的时候自动释放。

注意:虽然多个线性进入t1()都会创建一个新的lock_guard锁对象,但m全局是一份,所以多个线程走到lockGuard(m)全部都会阻塞,直到某个线程走完lockGuard(m)下面的部分,才会让一个新的线程进来,其他线程继续阻塞。

线程函数的参数怎么传递引用:

前两个传入线程函数的num是原num的一份拷贝,不是num本身,所以num本身的值不会变(第二种在VS里报错,如果线程函数参数是引用,传参也必须是引用传递)

线程函数是成员函数:

第一个参数为成员函数指针,需要加作用域, 第二个参数为对象指针,从第三个参数开始是成员函数参数传递。

注意:第一个参数可以把成员函数设为静态成员函数就不需要加作用域,直接传函数指针, 这样就没有第二个参数,但第三个参数需要传(静态成员函数实际是加了访问控制权限的全局函数,不属于类的对象)

thread m_thread = thread(do_some_work, 20);

线程锁需要注意的问题:

(1)线程锁可以设置静态或者全局变量,不管多线程创建几个对象,静态和全局都是一个,所有线程共享

(2)线程锁不能设成成员变量,因为一旦多线程同时创建多个对象时,线程锁就是多个,不是同一份

相关文章
|
1月前
|
缓存 安全 C++
C++无锁队列:解锁多线程编程新境界
【10月更文挑战第27天】
43 7
|
1月前
|
消息中间件 存储 安全
|
2月前
|
存储 消息中间件 资源调度
C++ 多线程之初识多线程
这篇文章介绍了C++多线程的基本概念,包括进程和线程的定义、并发的实现方式,以及如何在C++中创建和管理线程,包括使用`std::thread`库、线程的join和detach方法,并通过示例代码展示了如何创建和使用多线程。
55 1
C++ 多线程之初识多线程
|
2月前
|
存储 并行计算 安全
C++多线程应用
【10月更文挑战第29天】C++ 中的多线程应用广泛,常见场景包括并行计算、网络编程中的并发服务器和图形用户界面(GUI)应用。通过多线程可以显著提升计算速度和响应能力。示例代码展示了如何使用 `pthread` 库创建和管理线程。注意事项包括数据同步与互斥、线程间通信和线程安全的类设计,以确保程序的正确性和稳定性。
|
2月前
|
存储 前端开发 C++
C++ 多线程之带返回值的线程处理函数
这篇文章介绍了在C++中使用`async`函数、`packaged_task`和`promise`三种方法来创建带返回值的线程处理函数。
69 6
|
2月前
|
缓存 负载均衡 Java
c++写高性能的任务流线程池(万字详解!)
本文介绍了一种高性能的任务流线程池设计,涵盖多种优化机制。首先介绍了Work Steal机制,通过任务偷窃提高资源利用率。接着讨论了优先级任务,使不同优先级的任务得到合理调度。然后提出了缓存机制,通过环形缓存队列提升程序负载能力。Local Thread机制则通过预先创建线程减少创建和销毁线程的开销。Lock Free机制进一步减少了锁的竞争。容量动态调整机制根据任务负载动态调整线程数量。批量处理机制提高了任务处理效率。此外,还介绍了负载均衡、避免等待、预测优化、减少复制等策略。最后,任务组的设计便于管理和复用多任务。整体设计旨在提升线程池的性能和稳定性。
84 5
|
2月前
|
C++
C++ 多线程之线程管理函数
这篇文章介绍了C++中多线程编程的几个关键函数,包括获取线程ID的`get_id()`,延时函数`sleep_for()`,线程让步函数`yield()`,以及阻塞线程直到指定时间的`sleep_until()`。
30 0
C++ 多线程之线程管理函数
|
2月前
|
资源调度 Linux 调度
Linux C/C++之线程基础
这篇文章详细介绍了Linux下C/C++线程的基本概念、创建和管理线程的方法,以及线程同步的各种机制,并通过实例代码展示了线程同步技术的应用。
33 0
Linux C/C++之线程基础
|
4月前
|
Java 调度
基于C++11的线程池
基于C++11的线程池
|
4月前
|
Dart 编译器 API
Dart ffi 使用问题之在C++线程中无法直接调用Dart函数的问题如何解决
Dart ffi 使用问题之在C++线程中无法直接调用Dart函数的问题如何解决