Linux之线程及线程安全详解

news2024/11/24 3:12:20

前言:在操作系统中,进程是资源分配的基本单位,那么线程是什么呢?线程是调度的基本单位,我们该怎么理解呢?

目录

一,线程概念理解

二,Linux里面的线程原理

三,为什么要有线程

四,线程相关接口

1)线程创建

2)获取本线程ID

3)线程等待

4)线程取消

5)线程退出

五,多线程安全

1,互斥锁的原理

 2,互斥锁的使用

3,锁带来的饥饿问题

4,信号量

六,线程安全条件

1,常见的线程安全情况

2,常见的线程不安全情况

3,死锁


一,线程概念理解

现在我们举一个例子:我们以家庭为单位,将家庭看作一个整体,假如分配房子,车子等社会资源,这些资源被分配是以家庭为单位,这就类似于进程,每个进程执行一个大任务,进程之间的联系并不紧密并且相对独立。线程则类似与家庭里面的每一个人,家里面有许多为了完成大任务而拆分出来的小任务,比如孩子要上学,父母要上班,还有家务,家庭里面的每一个人有关联但也有各自不同的小任务,家庭里面的每一个人都共享所有资源,比如电视剧,客厅,车子——进程被分配的资源,但是同时它们也有自己的私人空间,线程同样如此,但线程的私人空间是

线程 ID
一组寄存器
errno
信号屏蔽字
调度优先级
为什么线程要有自己的私人空间呢?因为即使每个人都是为这个家庭(进程)做事,但是每个人做的事情不一样,我们需要一些私人数据才能完成不同的任务,而且为了区分不同的家庭成员(线程)我们也需要对他们进行起名(编号),这也是私人数据。

二,Linux里面的线程原理

大家有没有发现,线程和进程的功能其实有点类似,比如进程是执行一个复杂的大任务,而线程则是执行大任务里面拆分出来的小任务,并且它们都有自己的栈,共享进程的资源那我们可以使用进程的PCB来复用代替线程的TCB吗?答案是可以的,这样子不仅提高了代码的复用率,降低了编写的难度,让代码结构和维护变得更加简单,LinuxTCP的结构体就和PCB一样,但也可以自己编写一个独立的TCB,比如:windows系统。但也有所不同,进程号是标识进程唯一性的编号,而线程号则是一个地址——线程地址。
Linux里面的线程被称作轻量级进程,我们需要理清楚线程(轻量级进程)和进程之间的关系,进程是一组线程集合,一个进程最少有一个线程,线程则是进程里面的一个执行流(执行小任务)

三,为什么要有线程

大家可能很好奇为什么有进程了还要有线程,一个CPU一个时间段执行一个指令吗?线程虽称并行执行流,但底层还是不能同时执行,也得排队。

现在举一个例子:有一个进程A需要执行一个任务,从外设输入字符串并且打印到屏幕上,并且要计算一些加减乘除。

如果只有一个进程我们只能顺序执行,也就是当输入输出的时候我们不能干其他事必须等待,而IO流的速度是很慢的,如果我们一直等待不就把CPU的资源浪费了吗?如果我们利用线程呢?一个线程负责IO流,一个线程负责计算,这样子当我们进行IO操作的时候,我们可以把它挂起,利用CPU去进行计算,当IO操作完成再唤醒执行这个线程,这样子CPU的利用率就提高了,执行效率也提高了。

但是有人可能有疑问,我们为什么不切换到下一个进程,等到IO操作完成再唤醒这个进程呢?这就涉及到了开销的问题,进程之间是相互独立的,我们切换是需要进行保存现场等工作,这样子不断的切换开销是很大的,而线程它们之间的切换的开销小的很多,他们的页表,数据都是共享的。

那么线程适合什么样的场景使用,线程是越多越好吗?
线程适合IO操作较多的场景,计算流操作效果比较差,因为CPU的算力是有限的,同一时间只允许一个计算任务。

线程并不是越多越好,因为线程也是有开销的,例如TCB结构体,栈,对这些进行管理也要开销。

四,线程相关接口

在Linux里面实际上是没用线程的,只有轻量级进程,但是为了迎合主流,Linux还是对轻量级进程进行了包装成了线程库,因此在编译时要连接原生线程库,在g++指令后面加上  -lpthread,例如:

g++ -o test.o test.cpp -std=c++11 -lpthread
线程接口使用类似于进程间通信的接口,首先我们要认识一下区分线程的唯一标识符,也就是线程地址
pthread_t 

底层实际上就是一个unsigned int 

1)线程创建

 第一个参数thread会通过指针返回创建线程的ID,第二个参数大部分情况是NULL,用来调整线程的属性,第三个参数是线程执行的函数,第四个参数是执行函数的参数。成功返回0,失败返回错误编号。

void* task(void* arg){
int* i=(int*)arg;
cout<<"这是一个线程任务:"<<*i;
}
pthread_t tid;
int i=1;
pthread_create(&tid,NULL,task,(void*)i);
2)获取本线程ID

没有参数,返回值就是本线程的ID。

3)线程等待
第一个参数是等待线程ID,第二个是对线程进行管理的参数,一般默认NULL。
这个函数的作用是等待一个线程结束,成功返回0,失败返回错误编号,在这个线程结束前这个函数不会结束。
pthread_t tid=pthread_self();
pthread_join(tid,NULL);

4)线程取消
函数参数是取消线程的ID,操作成功返回0。
pthread_t tid=pthread_self();
pthread_cancel(tid);
5)线程退出

这个作用于pthread_exit作用效果类似,但是只能退出本线程,不能退出指定线程。

五,多线程安全

大家有没有想过多线程执行会不会带来安全问题,答案是必然的,为什么呢?因为线程简单切换会发生在如何不是原子代码执行的时候(原子性是指代码执行不会被中断,要么不开始,一开始就必须执行完,不存在中间状态),

int tictik=100;
void* RobTictik(void* arg){
while(titck>0){
cout<<"线程:"<<pthread_self()<<"抢到了票"<<tictik--<<endl;
}
}

如果所有线程执行这个函数,很有可能就会出现tictik最后小于0的情况,也就是最后卖出了多于100张的票,为什么呢?假如tictik已经是1了,线程A刚刚进入还未来得及打印将tictik打印就切换到了线程B,就会出现多卖票的情况。

那我们有什么办法解决吗?

1,互斥锁的原理

互斥锁是什么呢?人如其名它的功能类似于一把锁,你进去时候加上一把锁,当别人试图进来的时候就会因为没有钥匙而无法进来,你出去的是就把锁换回去,让其他想进来的人竞争这把钥匙。

锁的原理是什么呢?其实挺简单的,就是锁里面本来有一个1,当线程切换的时候线程会把自己的上下文保存,将数据1拿走,其他线程走到这块区域的时候就发现是0无法运行,继续等待抢锁,直到线程将这块区域运行完才会将锁换回去。下面这个图就是类似我讲述的锁原理。

 2,互斥锁的使用

互斥锁的使用需要初始化,然后加锁,解锁。

初始化有两种方式,一种是全局锁,一种是局部锁(作用域)

这是互斥锁的结构体

全局互斥锁初始化

局部互斥锁的初始化 

第一个参数是锁结构体,第二个参数一般填NULL。

加锁

成功返回0,需要注意的是加锁代码是原子性的,防止多个线程进入锁

解锁

成功返回0,注意解锁并不是原子性的,因为解锁时是不是原子性已经不重要了,如果锁已经归还,多线程也只能有一个抢到,如果还未归还不过是让其他线程多等等。

pthread_mutex_init(&_mutex,NULL);
 pthread_mutex_lock(&p->_mutex);
           //临界区代码,被保护,原子性 
 pthread_mutex_unlock(&p->_mutex);

3,锁带来的饥饿问题

互斥锁的抢夺是公平的,但是有一些线程的抢锁能力强,这就会导致一个问题,一个线程长期霸占着锁,其他线程就一直无法运行代码,导致饥饿问题,那有什么解决办法吗?答案是条件变量。

条件变量是什么呢?之前我们举例子所有人抢钥匙开门,现在我们加一个规矩,那就是排队,新来的和出去的只能从后面开始排队,而且这段时间你们都处于休眠,直到轮到你们有人唤醒你们才继续执行。

条件变量使用很类似于互斥锁

条件变量结构体

初始也分全局初始化和局部初始化

全局条件变量初始化

局部条件变量初始化

第一个参数是条件变量结构体,第二个参数一般是NULL。

互斥锁的使用一般是放在互斥锁里面的,如果将线程放入条件队列,会先解锁,然后继续抢锁,因此建议进入互斥锁临界区就先检查是否需要放入条件队列等待

参数一是条件变量结构体,参数二是互斥锁,因为条件变量是需要结合互斥锁使用的。

条件变量的唤醒,我们直到进入条件变量等待队列后是无法自己醒来的,需要使用函数唤醒

唤醒指定条件变量里面的一个线程,成功返回0

唤醒指定条件变量里面的所有线程,成功返回0

破坏条件变量

pthread_mutex_init(&_mutex,NULL);
pthread_cond_init(&cond,NULL);
 pthread_mutex_lock(&p->_mutex);
while(条件不满足){
 pthread_cond_wait(&cond,&mutex);
}
           //临界区代码,被保护,原子性 
 pthread_mutex_unlock(&p->_mutex);

上面的代码为什么要用循环来判断条件是否满足呢?因为即使抢到锁了条件也不一定满足,如果是if语句就会直接执行接下里的代码,导致线程安全问题 

4,信号量

在Linux里面信号量也是保护线程安全的一种重要手段,一般也是结合互斥锁使用

信号量的原理就是计数器,但是对计数器的操作是原子性的,举个例子,假如盆里面有十个苹果,有三个人都想抢苹果,三个让可以同时拿苹果,但是不能抢同一个苹果,信号量就是保护你们不抢同一个苹果。

信号量结构体

信号量的初始化只有一种

第二个参数一般设置为0,第三个参数是sem量的初始值,类似于上面的盆子里有几个苹果,成功返回0。

申请信号量,也就是上面申请抢一个苹果,成功返回0.

释放信号量,相当于有人往盆里放苹果,成功返回0。

六,线程安全条件

什么样的线程有风险,什么样的线程是安全的呢?

1,常见的线程安全情况

只读不写

执行流里面的写操作都是原子性的

多个线程切换不存在二义性

2,常见的线程不安全情况

不保护多线程共享的变量

执行流的状态随着执行,被调用状态发生变化

返回指向静态变量的函数

调用线程不安全的函数

3,死锁

死锁是指各自不释放自己占有有资源,但因为有资源抢夺不到而都无法导致一种尴尬的场景。举个例子,想要打开一个宝箱需要两个要是,有两个人各自持有一把锁(线程各自持有一个锁),双方互不相让,导致谁也打不开宝箱,死锁和多个锁之间分配顺序的不同有很大关系。

死锁但是有四个必要的条件

1,不可剥夺性,线程占有资源互不相让,别人无法强行抢夺自己以有的资源

2,互斥条件 ,一个资源不能同时被多个人使用

3,请求和保持条件,一个执行流因请求资源而阻塞时,对已获得的资源保持不放

4,循环等待条件,形成了环路,造成了尴尬的场面,谁也无法好过。

如何避免死锁

破坏上面的四个形成的必要条件之一,死锁就不攻自破

加锁顺序一致,防止各自持有对方所需的资源

避免锁未释放,资源被锁死

资源一次性释放

银行家算法:模拟资源分配,如果产生了死锁就撤销任务不分配资源

死锁检测算法

拓展:C++里面的各种STL容器为了追求效率是没用加锁的,使用的时候要注意线程安全。

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

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

相关文章

哈夫曼树的构造,哈夫曼树的存在意义--求哈夫曼编码

一:哈夫曼树的构造 ①权值,带权路径长度。 ②一组确定权值的叶子节点可以构造多个不同的二叉树,但是带权路径长度min的是哈夫曼树 ③算法基本思想及其实操图片演示 注:存储结构和伪代码 1 初始化: 构造2n-1棵只有一个根节点的二叉树,parent=rchild=lchild=-1; 其中…

忆恒创源国产系列新品 —— PBlaze7 7A40 取得 PCI-SIG 兼容性认证

在此前报道中&#xff0c;我们曾预告了忆恒创源国产系列 PCIe 5.0 SSD 新品 —— PBlaze7 7A40&#xff0c;今天&#xff0c;这款 SSD 已经顺利通过 PCI-SIG 的严格测试并出现在 Integrators List 集成商列表当中&#xff0c;标志着距离 PBlaze7 7A40 的正式发布又近了一步。 正…

Spring Boot框架基础

文章目录 1 Spring Boot概述2 Spring Boot入门2.1 项目搭建2.2 入门程序 3 数据请求与响应3.1 数据请求3.2 数据响应 4 分层解耦4.1 三层架构4.2 控制反转4.3 依赖注入 5 参考资料 1 Spring Boot概述 Spring是Java EE编程领域的一个轻量级开源框架&#xff0c;是为了解决企业级…

乐高小人分类项目

数据来源 LEGO Minifigures | Kaggle 建立文件目录 BASE_DIR lego/star-wars-images/ names [YODA, LUKE SKYWALKER, R2-D2, MACE WINDU, GENERAL GRIEVOUS ] tf.random.set_seed(1)# Read information about dataset if not os.path.isdir(BASE_DIR train/):for name in …

Edge 工作区是什么?它都有哪些作用?

什么是工作区 Edge 工作区是什么&#xff1f;它是微软 Edge 浏览器中的一个功能&#xff0c;在帮助用户更好地组织和管理他们的浏览会话。通过工作区&#xff0c;用户可以创建多个独立的浏览环境&#xff0c;每个工作区内包含一组相关的标签页和浏览器设置。这使得用户能够根据…

asp.net core使用httpclient

主要讲解常见的get请求和post请求 GET var client new HttpClient(); //3秒钟不响应就超时 client.TimeoutTimeSpan.FromSeconds(3); using HttpResponseMessage response await client.GetAsync("todos/3"); var jsonResponse await response.Content.ReadAsSt…

变压器绕线完成之后要做的事

1 调整感量&#xff1a;测主绕组电感量&#xff0c;通过磨气隙或垫气隙&#xff0c;测得感量没错以后&#xff0c;用胶带封装磁芯 2 测验同名端是否正确&#xff1a;两绕组首尾相连&#xff0c;测试连接后的总感量&#xff0c;是否比感量大的那个绕组还大。如果是&#xff0c;…

Allegro热风焊盘制作教程

阿里狗热风焊盘制作教程 打开PCB Editor&#xff0c;新建Flash symbol&#xff0c;最好保存在与Pad文件同一个路径 点击Setup–>Design Parameter Editor,设置mm单位&#xff0c;在设置画布&#xff0c;把原点提上去&#xff0c;点击Apply和OK 把视野调整到原点 点击Setup–…

vue面试题2-根据以下问题回答

以下是针对提供的关于Vue的问题的回答&#xff1a; Vue的基本原理&#xff1a; Vue.js是一个流行的JavaScript框架&#xff0c;用于构建用户界面和单页面应用。其基本原理包括响应式数据、模板、组件系统、指令、生命周期钩子和虚拟DOM。 双向数据绑定的原理&#xff1a; Vue通…

咖啡机器人如何精准控制液位流量

在如今快节奏的生活中&#xff0c;精确控制液位流量的需求愈发迫切&#xff0c;特别是在咖啡机器人等精密设备中。为了满足这一需求&#xff0c;工程师们不断研发出各种先进的技术&#xff0c;以确保液体流量的精准控制。其中&#xff0c;霍尔式流量计和光电式流量计就是两种常…

转让北京海淀成立满1年拍卖公司许可证条件和流程

拍卖经营批准证书是拍卖企业经营所需的许可&#xff0c;是为了维护拍卖秩序&#xff0c;保护拍卖活动各方合法权益而颁发的合法凭证。其中个人物品&#xff0c;公司物品&#xff0c;或者国有资源的拍卖可通过普通拍卖资质进行拍卖。而文物古董拍卖类的需取得文物拍卖经营许可证…

google keybox.xml格式 内容有哪些 Keybox数量、设备ID、算法的 私钥 公钥 证书链 (ECDSA即ECC, RSA)

根据您提供的文件内容&#xff0c;keybox.xml 文件包含以下主要信息&#xff1a; Keybox数量 ([NumberOfKeyboxes](file:///d%3A/010F200/svn/ProduceToolMfc/FtSmartPos/FtSmartPos/ToolBydMes/httpclient/e%3A%5CGoogleKey%5CLinux_AttestationKeyboxPack_Tool%5CLinux_Atte…

【吊打面试官系列】简述在 MySQL 数据库中 MyISAM 和 InnoDB 的区别 ?

大家好&#xff0c;我是锋哥。今天分享关于 【简述在 MySQL 数据库中 MyISAM 和 InnoDB 的区别 &#xff1f;】面试题&#xff0c;希望对大家有帮助&#xff1b; 简述在 MySQL 数据库中 MyISAM 和 InnoDB 的区别 &#xff1f; MyISAM&#xff1a; 不支持事务&#xff0c;但是…

全网最全!场外个股期权的询价下单流程的详细解析

场外个股期权的询价下单流程 场外个股期权交易&#xff0c;作为在交易所外进行的个性化期权交易方式&#xff0c;为投资者提供了更加灵活和定制化的交易选择。以下是场外个股期权询价下单流程的详细步骤&#xff1a; 文章来源/&#xff1a;财智财经 第一步&#xff1a;明确交…

APP兼容性测试都需要考虑哪些场景?

APP测试的时候都需要验证兼容性。那兼容性测试需要考虑哪些场景&#xff1f; 进行APP的兼容性测试时&#xff0c;需要考虑以下一些常见的测试场景&#xff1a; 1. 操作系统兼容性&#xff1a;测试应用程序在不同操作系统上的兼容性&#xff0c;如iOS、Android、Windows等。确…

一文搞懂DevOps、DataOps、MLOps、AIOps:所有“Ops”的比较

引言 近年来&#xff0c;“Ops”一词在 IT 运维领域的使用迅速增加。IT 运维正在向自动化过程转变&#xff0c;以改善客户交付。传统的应用程序开发采用 DevOps 实施持续集成&#xff08;CI&#xff09;和持续部署&#xff08;CD&#xff09;。但对于数据密集型的机器学习和人…

量水堰计的校准和误差分析:提高测量精度的技巧

量水堰计作为一种重要的流量测量工具&#xff0c;广泛应用于水利工程、环境监测和农业灌溉等领域。为了确保其测量结果的准确性和可靠性&#xff0c;对量水堰计进行定期校准和误差分析至关重要。本文将详细讨论量水堰计的校准方法、误差来源以及提高测量精度的技巧。 一、量水堰…

积极乐观的心态对学习和研究机器人相关技术非常重要

微案例 把难换成有趣&#xff0c;动力就会足一些。 或者把难换成“有挑战”等。 负面情绪描述&#xff1a; 学习和研究机器人很难。 中性情绪描述&#xff1a; 学习和研究机器人有挑战。 正面情绪描述&#xff1a; 学习和研究机器人很有趣。 机器人专业不合格且失败讲师如何让…

揭秘Lazada API:掌握数据驱动的电商帝国,轻松实现销售飞跃

当涉及到Lazada API接口的技术帖子时&#xff0c;通常我们需要了解如何与Lazada的API进行交互&#xff0c;以执行各种操作&#xff0c;如获取产品信息、处理订单、管理库存等。由于Lazada的API是私有的并且需要特定的认证和访问权限&#xff0c;以下是一个简化的示例&#xff0…

视频、图片、音频资源抓取(支持视频号),免安装,可批量,双端可用!

今天分享一款比较好用资源嗅探软件&#xff0c;这个嗅探工具可以下载视频号&#xff0c;界面干净&#xff0c;可以内容预览和批量下载&#xff0c;看到这里你是不是想用它爬很多不得了的东西。这款软件无需安装&#xff0c;打开即用。同时他支持windows系统和Mac系统,是一款不可…