C++线程操作
admin
2024-02-26 21:01:02
0

文章目录

  • 一、线程和进程的基本概念
  • 二、C++中的线程
  • 三、互斥量mutex的使用
  • 四、条件变量condition_variable的使用
  • 五、线程的其他使用


一、线程和进程的基本概念

 
线程是操作系统进行运算调度的基本单位,进程是操作系统进行资源分配的基本单位。
不同进程分配不同的内存区域,同一个进程中的不同线程共享同一片内存。
 
线程的不同状态:

  1. 新建状态:仅仅是语言层面创建了线程状态,但是还没有与操作系统线程进行关联;
  2. 就绪状态:线程已经与操作系统相关联,可以被CPU进行调度;
  3. 运行状态:线程正占用着CPU,正在运算;
  4. 阻塞状态:等待被唤醒的状态;
  5. 终止状态:线程运行结束。

并行(Parallel)是真正意义上的同时运行,在同一时间点有多个程序在同时运行,比如多核CPU多个核同时运行着不同的程序。并发(ConCurrent)是在同一个时间段上多个程序同时运行,在宏观上看是同时运行,微观上是交替运行。
 
之所以设计多线程是为了充分的利用CPU资源,比如线程A正在等待磁盘数据的时候,可以将CPU让出来,线程B就能占用CPU进行计算。
 

二、C++中的线程

 
c++11新增了与操作系统无关的线程类thread,使用方法如下:

void func(int i) {cout << i << endl;
}int main() {thread t1(func, 10);  //传入线程绑定的函数和函数所需参数t1.join();return 0;
}

thread对象创建之后,线程就已经进入就绪态,可以占用CPU进行运算。
线程的运行需要以具体的函数作为入口,而函数的参数传递无非值传递,引用传递和指针传递3种形式,值传递和指针传递没什么好说的,直接复制值或者指针放入线程入口函数中,而线程的使用中需要特别注意⚠️⚠️⚠️引用传递,普通函数使用引用传递,底层使用的还是指针,所以函数中的引用会关联到函数外的变量。但是在线程的使用中,函数引用传递是引用的复制,函数中的引用和函数外的值没有关联关系。
 
join()detach()的用法:调用线程的join()方法意味着,主线程会阻塞在该语句处,等待子线程运行结束的时候才会继续运行主线程,而detach()方法意味着,子线程从主线程中分离出来各自运行,子线程变为后台守护线程(deamon thread),如此这般,子线程可能会结束在主线程之后,可能会出现不可预知的问题。
 

三、互斥量mutex的使用

 
当存在多个线程共同访问同一资源(比如队列)的时候,如果操作的顺序不当可能会出现不可预知的错误,该资源成为临界资源,此时C++使用互斥量mutex进行资源访问的控制。
 
通常为了方便理解多线程操作,我们将mutex称为锁🔒,对资源的访问称为对资源上锁和解锁。实际上mutex是一个标记量,不同的线程对同一mutex对象进行lock()unlock()可以达到控制资源访问次序的目的。
 
lock()方法是尝试获取锁并加锁,如果获取不到锁,线程就会阻塞在此语句处,mutex调用了lock()就必须unlock(),否则就有可能是其他线程不能成功lock()而一直阻塞。
为了解决这种可能忘记unlock()的问题,引入lock_guard模版类,lock_guard类似于智能指针,出作用域会自动unlock()。和lock_guard相似的一个模版类是unique_lock,使用方法和lock_guard类似。
 

四、条件变量condition_variable的使用

 
多个线程需要通信的时候会用到cv,比如消费者和生产者线程模型,这两个线程并不是单单的对资源解锁和加锁。而是需要通信,比如,消费者没有资源可以消费的时候需要将生产者叫醒。
 
cv有两个方法wait()wait_for()wait方法会让当前线程释放锁然后阻塞,直到被唤醒;wait_for方法需要传递一个时间参数,当前线程释放锁然后等待被唤醒,但是不会一直等待,而是吵过时间参数自动苏醒。
cv还有两个方法notify_one()notify_all()notify_one会唤醒线程等待队列的第一个线程,而notify_all会唤醒所有等待的线程。
 
waitsleep方法的区别:

  1. waitcv的方法,sleepthis_thread的方法;
  2. wait会释放锁然后再阻塞,而sleep直接阻塞,不会释放锁。

 

五、线程的其他使用

  1. wait被唤醒之后第一件事是会反复尝试直到拿到锁再往下执行,wait被唤醒拿到锁之后一般还要再次检查阻塞条件保证多线程共享的正确性;
  2. unique_lock所有权的转移类似于unique_ptr,使用move进行所有权转移;
  3. std::lock(mutex1, mutex2)方法可以尝试同时锁住多个互斥量,如果有一个没锁成功,就不会对任何互斥量进行加锁;
  4. release()方法将unique_lockmutex解绑,并返回之前绑定的mutex
  5. 互斥量mutexlock()unlock()之间包裹的语句多少称为锁的粒度,粒度要合适,如果太小可能不能达到上锁的预期,如果太大,会降低多线程运行效率;
  6. std::adopt_lock可以传递给lock_guardunique_lock的构造函数,表示这个互斥量已经被lock了(你必须要把互斥量提前lock了 ,否者会报异常);std::adopt_lock标记的效果就是假设调用一方已经拥有了互斥量的所有权(已经lock成功了);通知lock_guard不需要再构造函数中lock这个互斥量了。unique_lock也可以带std::adopt_lock标记,含义相同,就是不希望再unique_lock的构造函数中lock这个mutex。用std::adopt_lock的前提是,自己需要先把mutex lock上;用法与lock_guard相同;
  7. std::try_to_lock会尝试用mutex的lock去锁定这个mutex,但如果没有锁定成功,也会立即返回,并不会阻塞在那里,用这个try_to_lock的前提是你自己不能先lock;
  8. std::defer_lock的前提是,你不能自己先lock,否则会报异常,std::defer_lock的意思就是并没有给mutex加锁,初始化了一个没有加锁的mutex

相关内容

热门资讯

【MySQL】锁 锁 文章目录锁全局锁表级锁表锁元数据锁(MDL)意向锁AUTO-INC锁...
【内网安全】 隧道搭建穿透上线... 文章目录内网穿透-Ngrok-入门-上线1、服务端配置:2、客户端连接服务端ÿ...
GCN的几种模型复现笔记 引言 本篇笔记紧接上文,主要是上一篇看写了快2w字,再去接入代码感觉有点...
数据分页展示逻辑 import java.util.Arrays;import java.util.List;impo...
Redis为什么选择单线程?R... 目录专栏导读一、Redis版本迭代二、Redis4.0之前为什么一直采用单线程?三、R...
【已解决】ERROR: Cou... 正确指令: pip install pyyaml
关于测试,我发现了哪些新大陆 关于测试 平常也只是听说过一些关于测试的术语,但并没有使用过测试工具。偶然看到编程老师...
Lock 接口解读 前置知识点Synchronized synchronized 是 Java 中的关键字,...
Win7 专业版安装中文包、汉... 参考资料:http://www.metsky.com/archives/350.htm...
3 ROS1通讯编程提高(1) 3 ROS1通讯编程提高3.1 使用VS Code编译ROS13.1.1 VS Code的安装和配置...
大模型未来趋势 大模型是人工智能领域的重要发展趋势之一,未来有着广阔的应用前景和发展空间。以下是大模型未来的趋势和展...
python实战应用讲解-【n... 目录 如何在Python中计算残余的平方和 方法1:使用其Base公式 方法2:使用statsmod...
学习u-boot 需要了解的m... 一、常用函数 1. origin 函数 origin 函数的返回值就是变量来源。使用格式如下...
常用python爬虫库介绍与简... 通用 urllib -网络库(stdlib)。 requests -网络库。 grab – 网络库&...
药品批准文号查询|药融云-中国... 药品批文是国家食品药品监督管理局(NMPA)对药品的审评和批准的证明文件...
【2023-03-22】SRS... 【2023-03-22】SRS推流搭配FFmpeg实现目标检测 说明: 外侧测试使用SRS播放器测...
有限元三角形单元的等效节点力 文章目录前言一、重新复习一下有限元三角形单元的理论1、三角形单元的形函数(Nÿ...
初级算法-哈希表 主要记录算法和数据结构学习笔记,新的一年更上一层楼! 初级算法-哈希表...
进程间通信【Linux】 1. 进程间通信 1.1 什么是进程间通信 在 Linux 系统中,进程间通信...
【Docker】P3 Dock... Docker数据卷、宿主机与挂载数据卷的概念及作用挂载宿主机配置数据卷挂载操作示例一个容器挂载多个目...