音视频学习笔记——c++多线程(一)

✊✊✊🌈大家好!本篇文章主要整理了部分多线程相关的内容重点😇。首先讲解了多进程和多线程并发的区别以及各自优缺点,之后讲解了Thead线程库的基本使用。


本专栏知识点是通过<零声教育>的音视频流媒体高级开发课程进行系统学习,梳理总结后写下文章,对音视频相关内容感兴趣的读者,可以点击观看课程网址:零声教育


🎡导航小助手🎡

    • 一、多进程与多线程
    • 二、Thead线程库的基本使用
    • 三、小结

一、多进程与多线程

首先有一个直观的理解:
  1.进程就是运行中的程序
  2.线程就是进程中的进程

操作系统中可以有多个进程,一个进程中也可以有多个线程。

1.1 多线程并发
多进程并发是将一个应用程序划分为多个独立的进程(每个进程只有一个线程),这些独立的进程之间相互可以通信,共同完成任务。操作系统常常对进程提供大量的保护机制避免出现一个进程修改其他进程的数据,因此,相对于多线程,使用多进程更容易写出相对安全的代码。但这也造成了多进程并发存在两个不足之处

  1. 进程间的通信,无论是使用信号、套接字、还是文件、管道等方式,其使用要么比较复杂,要么就是速度较慢或者两者兼而有之。
  2. 运行多个线程,需要操作系统花费很多资源进行管理。

在多个进程并发完成一个任务时,常会出现操作同一个数据以及进程之间的相互通信,因此,多进程并发不是一个很好的选择。
1.2 多线程并发
多线程并发:同一个进程中执行多个线程。

  • 优点:线程是轻量级的进程,每个线程可以独立的运行不同的指令序列,且线程不独立拥有资源,依赖于创建它的进程而存在。
    同一进程中的多个线程能够很方便的进行数据共享以及通信,比进程更适用于并发操作。
    不足:缺少操作系统提供的保护机制。在多线程共享数据及通信时,需程序员做更多的操作,并且还需极力避免死锁

二、Thead线程库的基本使用

2.1 创建线程
要创建线程,我们需要一个可调用函数或函数对象,作为线程的入口点。
在C++11中,我们可以使用函数指针、函数对象lambda表达式来实现。
创建线程的基本语法如下:

#include <thread>std::thread t(function_name, args...);
  • function_name是线程入口点的函数或可调用对象
  • args...是传递给函数的参数
    创建线程后,我们可以使用t.join()等待线程完成,或者使用t.detach()分离线程,让它在后台运行。
    实例1:
	#include <iostream>
	#include <thread>
	using namespace std;
	void doit() { cout << "World!" << endl; }
	int main() {
		thread a([] {
			cout << "Hello, " << flush;
			}), b(doit);
			a.join();
			b.join();
			return 0;
	}

运行结果:
在这里插入图片描述
在这里插入图片描述
上面两次结果并不相同,这是因为多线程运行时是以异步方式执行的,与我们平时写的同步方式不同。异步方式可以同时执行多条语句谁先执行得快,谁先执行完
实例2:

	#include <iostream>
	#include <thread>
	using namespace std;
	void thread_1(){
		cout<<"线程t"<<endl;
	}

	void print_message(const string& message) {    
		cout << message <<endl;
		cout << "线程 t" << endl;
	}
	void increment(int& x) {   
		++x;
		cout << "线程 t2" << endl;
	}
	int main() {  
		cout << "主线程1\n";
		thread t(thread_1);// 开启线程t,调用:thread_1()
		t.join();
		cout << "子线程t结束\n";
		string message = "Hello, world!";    
		thread t1(print_message, message);// 开启线程t1,调用:print_message()
		t1.join();   
		 
		int x = 0;   
		thread t2(increment, ref(x));//开启线程t1,调用:increment() 
		t2.join(); 
		cout << "子线程t2结束\n";
		cout << x << endl;    
		cout << "全部子进程结束\n";
		return 0;
	}

运行结果:
在这里插入图片描述
从上面结果,我们很明显能看出,使用t.join()后程序需要等待进程t结束后,才会接着进行。
注意:thread在传递参数时,是以右值传递的。
我们在传递引用的时候,需要用到std::ref和std::cref

  • std::ref 可以包装按引用传递的值。
  • std::cref 可以包装按const引用传递的值。

2.2 join与detach方式
当线程启动后,一定要在和线程相关联的thread销毁前,确定以何种方式等待线程执行结束。比如上
例中的join。
detach方式,启动的线程自主在后台运行,当前的代码继续往下执行,不等待新线程结束。
join方式,等待启动的线程完成,才会继续往下执行。
可以使用joinable判断是join模式还是detach模式。

示例1:join举例

	#include <iostream>
	#include <thread>
	using namespace std;
	void thread_1() {
		while (1) {
			cout<<"子线程1"<<endl;
		}
	}
	void thread_2(int x) {
		while (1) {
			cout<<"子线程2"<<endl;
		}
	}
	int main() {
		thread first(thread_1);
		// 开启线程,调用:thread_1()
		thread second(thread_2, 100);
		// 开启线程,调用:thread_2(100)
		first.join(); // pauses until first finishes 这个操作完了之后才能destroyed
		second.join(); // pauses until second finishes//join完了之后,才能往下执行。
		while (1) {
			std::cout << "主线程\n";
		}
		return 0;
	}

在这里插入图片描述
线程1和线程2写的是死循环,那么在两个子线程没结束前,主线程不会执行。

示例2:detach举例

	#include <iostream>
	#include <thread>
	using namespace std;
	void thread_1() {
		while (1) {
			cout<<"子线程1"<<endl;
		}
	}
	void thread_2(int x) {
		while (1) {
			cout<<"子线程2"<<endl;
		}
	}
	int main() {
		thread first(thread_1);
		// 开启线程,调用:thread_1()
		thread second(thread_2, 100);
		// 开启线程,调用:thread_2(100)
		first.join(); // pauses until first finishes 这个操作完了之后才能destroyed
		second.join(); // pauses until second finishes//join完了之后,才能往下执行。
		while (1) {
			std::cout << "主线程\n";
		}
		return 0;
	}

运行结果:
在这里插入图片描述
在这里插入图片描述可以看出,主线程不会等待子线程1和2结束。如果主线程运行结束,程序则结束。
2.3 joinable
joinable()返回一个bool值,判断是join模式还是detach模式。
使用方法;

	if (myThread.joinable()) 1 foo.join();

三、小结

  1. 多进程安全但是浪费操作系统资源且进程间相互通信比较麻烦。多线程则可以很好的处理这两个问题,但是使用时需要使用更多操作确保安全。
  2. C++11提供了语言层面上的多线程,包含在头文件中。它解决了跨平台的问题,提供了管理线程、保护共享数据、线程间同步操作、原子操作等类。主要讲解了Thead线程库的基本使用,包括join()、joinable()和detach(),并举了很多例子进行补充。

感谢大家阅读!
接下来还会继续更新多线程相关知识,感兴趣的可以看其他笔记!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

react的diff源码

react 的 render 阶段&#xff0c;其中 begin 时会调用 reconcileChildren 函数&#xff0c; reconcileChildren 中做的事情就是 react 知名的 diff 过程 diff 算法介绍 react 的每次更新&#xff0c;都会将新的 ReactElement 内容与旧的 fiber 树作对比&#xff0c;比较出它们…

消息队列-kafka-消息发送流程(源码跟踪) 与消息可靠性

官方网址 源码&#xff1a;https://kafka.apache.org/downloads 快速开始&#xff1a;https://kafka.apache.org/documentation/#gettingStarted springcloud整合 发送消息流程 主线程&#xff1a;主线程只负责组织消息&#xff0c;如果是同步发送会阻塞&#xff0c;如果是异…

【CSP试题回顾】202104-2-邻域均值

CSP-202104-2-邻域均值 关键点&#xff1a;二维差分数组 详见&#xff1a;【CSP考点回顾】差分数组 解题思路 初始化矩阵和参数&#xff1a;首先&#xff0c;代码接收矩阵的大小&#xff08;n x n&#xff09;&#xff0c;每个元素的亮度值&#xff08;位于[0, L]区间&…

基于Vue的体育汇App设计与实现

目 录 摘 要 I Abstract II 引 言 1 1 核心技术的理论与分析 3 1.1 客户端技术 3 1.1.1 Vue.js框架 3 1.1.2 Vue.js路由管理 3 1.1.3 Vuex状态管理 3 1.1.4 MVVM开发模式 4 1.1.5 Vant组件库 5 1.2 服务端技术 5 1.2.1 Node.js 5 1.2.2 Egg.js框架 5 1.3 数据库技术 6 1.4 本章…

webUI自动化测试框架

&#x1f525; 交流讨论&#xff1a;欢迎加入我们一起学习&#xff01; &#x1f525; 资源分享&#xff1a;耗时200小时精选的「软件测试」资料包 &#x1f525; 教程推荐&#xff1a;火遍全网的《软件测试》教程 &#x1f4e2;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1…

【LeetCode】升级打怪之路 Day 16:二叉树题型 —— 二叉树的构造

今日题目&#xff1a; 654. 最大二叉树105. 从前序与中序遍历序列构造二叉树106. 从中序与后序遍历序列构造二叉树889. 根据前序和后序遍历构造二叉树 目录 LC 654. 最大二叉树 【easy】 Problem&#xff1a;根据遍历序列来还原二叉树 【classic】 ⭐⭐⭐⭐⭐LC 105. 从前序与中…

数据库原理实验课(1)

目录 实验内容 安装头歌中的相关内容 具体过程 完结撒花~ 我也是第一次接触oracle的相关软件和操作&#xff0c;所以是一次傻瓜式教学记录 实验内容 安装头歌中的相关内容 具体过程 这是我在百度网盘中下载解压出来的oracle文件夹内的全部内容&#xff08;可能有因为安装完…

使用Portainer让测试环境搭建飞起来

Docker的用处不多加赘述&#xff0c;Docker目前有以下应用场景&#xff1a; 测试&#xff1a;Docker很适合用于测试发布&#xff0c;将 Docker 封装后可以直接提供给测试人员进行运行&#xff0c;不再需要测试人员与运维、开发进行配合&#xff0c;进行环境搭建与部署。 测试…

【技术】基于Github Pages搭建个人博客静态网页

写在前面&#xff1a; 如果文章对你有帮助&#xff0c;记得点赞关注加收藏一波&#xff0c;利于以后需要的时候复习&#xff0c;多谢支持&#xff01; 文章目录 一、技术基础二、新建特殊仓库三、上传网页文件四、Github Pages设置 个人网页作为仅服务个人的网页&#xff0c;一…

Grafana变量默认全选

注&#xff1a;本文基于Grafana v9.2.8编写 1 问题 EKS集群里的node按照不同label被分为几类&#xff0c;我需要对这几类的node做一些统计。我希望当我使用lable选择时&#xff0c;node的值自动设置为该lable的所有node集合&#xff0c;而不需要再手动全选。 2 解决方案 变…

微信小程序(五十四)腾讯位置服务示范(2024/3/8更新)

教程如下&#xff1a; 上一篇 1.先在官网注册一下账号&#xff08;该绑定的都绑定一下&#xff09; 腾讯位置服务官网 2.进入控制台 3.创建应用 3. 额度分配 4.下载微信小程序SDK 微信小程序SDK下载渠道 5.解压将俩js文件放在项目合适的地方 6.加入安全域名or设置不验证合…

扩展CArray类,增加Contain函数

CArray不包含查找类的函数&#xff0c;使用不便。考虑扩展CArray类&#xff0c;增加Contain函数&#xff0c;通过回调函数暴露数组元素的比较方法&#xff0c;由外部定义。该方法相对重载数组元素的“”符号更加灵活&#xff0c;可以根据需要配置不同的回调函数进行比较 //类型…

AD20中关于“failed to add class member”的解决方法

目录 问题描述&#xff1a; 解决方法&#xff1a; 1.切换至PCB界面-选择工具栏的设计-类 2.把Component classes中的所有的类全部按照图中删除&#xff0c;保存 3.重新返回原理图界面转换PCB即可成功 问题描述&#xff1a; failed to add class member&#xff1a;未能添加…

解答关于:水牛社软件是做什么的?水牛社软件靠谱么?

很多对我们软件感兴趣但是没有入手的观望者都会有这样的疑问&#xff1a;水牛社软件具体是做什么的&#xff1f;水牛社软件靠谱么&#xff1f; 其实软件的简介已经讲的很清楚了&#xff0c;但是软件不是功能性软件&#xff0c;所以不能给大家免费试用&#xff0c;短期任务版块…

智能驾驶规划控制理论学习08-自动驾驶控制模块(轨迹跟踪)

目录 一、基于几何的轨迹跟踪方法 1、基本思想 2、纯追踪 3、Stanly Method 二、PID控制器 三、LQR&#xff08;Linear Quadratic Regulator&#xff09; 1、基本思想 2、LQR解法 3、案例学习 基于LQR的路径跟踪 基于LQR的速度跟踪 4、MPC&#xff08;Mode…

Python通过SFTP实现网络设备配置备份

一、背景 为了防止网络设备意外损坏&#xff0c;导致配置文件无法恢复&#xff0c;可以通过将网络设备的配置文件备份到本地电脑上。 一般情况下&#xff0c;设备支持通过FTP、TFTP、FTPS、SFTP和SCP备份配置文件。其中使用FTP和TFTP备份配置文件比较简单&#xff0c;但是存在…

JAVA虚拟机实战篇之内存调优[4](内存溢出问题案例)

文章目录 版权声明修复问题内存溢出问题分类 分页查询文章接口的内存溢出问题背景解决思路问题根源解决思路 Mybatis导致的内存溢出问题背景问题根源解决思路 导出大文件内存溢出问题背景问题根源解决思路 ThreadLocal占用大量内存问题背景问题根源解决思路 文章内容审核接口的…

尚硅谷JavaScript高级学习笔记

01 准备 JavaScript中函数是对象。我们后续描述构造函数的内存模型时&#xff0c;会将构造函数称为构造函数对象。 02 数据类型 typeof 运算符来查看值的类型&#xff0c;它返回的是类型的字符串值 会做数据转换 03 相关问题 04数据_变量_内存 05相关问题1 06相关问题2 …

办公电脑换成MacBookPro半年之后……

小白是从2008年开始接触电脑的&#xff0c;当时朋友给我注册的第一个QQ账号是2008年4月。 从此&#xff0c;小白一直认为电脑全部都是Windows系统。直到上大学那年&#xff0c;看到了外教老师的MacBookPro…… 折腾电脑的开始居然是起源于诺基亚手机&#xff0c;给半智能S40的…

Igraph入门指南 3

4、图转换到其他R数据结构 图是对实体关系的表达&#xff0c;在igraph中&#xff0c;图可以转换为三种数据结构。 4-1 图转邻接矩阵&#xff1a;as_adjacency_matrix | as_adj&#xff0c;结果是矩阵 邻接矩阵又分为有向图邻接矩阵和无向图邻接矩阵&#xff0c;但本函数使用…