Linux学习第18天:Linux并发与竞争: 没有规矩不成方圆

2023-11-02 22:10

本文主要是介绍Linux学习第18天:Linux并发与竞争: 没有规矩不成方圆,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Linux版本号4.1.15   芯片I.MX6ULL                                    大叔学Linux    品人间百味  思文短情长


        提到锁”,可能想到的更多的是限制。现实中,生活中锁也 存在于身边的方方面面。正所谓没有规矩不成方圆, 没有身边的这些锁,这些限制,社会将会变得无序、混乱。为了规范这些无序和混乱,就得根据实际情况制定规则制度甚至法律来进行束缚和限制。正如本篇笔记要讲解的内容一样,Linux内核采用一定的方式方法(函数)制定了这些所谓的规则,才能使程序变得更流畅。

        本篇笔记主要学习Linux处理并发与竞争的机制。主要内容包括原子操作、自旋锁、信号量和互斥体。

一、并发与竞争

1.简介

        Linux是多任务操作系统,存在多个任务同时访问同一内存区域,造成这些任务会相互覆盖这段内存中的数据,从而造成内存数据的混乱。多个线程同时操作临界区就会发生竞争(竞争)并发导致。

2.保护什么

        保护共享资源(数据)。

二、原子操作

1.简介

        原子操作就是指不能再进一步分割的操作,用于变量和位操作。

2.原子整形操作API函数

原子整形操作API函数
函数描述
ATOMIC_INIT(init i)定义原子变量的时候对其进行初始化。
int atomic_read(atomic_t *v)读取v的值,并返回。
void atomic_set(atomic_t *v,int i)向v写入i值。
void atomic_add(atomic_t *v)给v加上i值。
void atomic_sub(atomic_t *v)给v减去i值。
void atomic_inc(atomic_t *v)自增
void atomic_dec(atomic_t *v)自减
void atomic_inc_return(atomic_t *v)自增并返回v值
void atomic_dec_return(atomic_t *v)自减并返回v值
int atomic_sub_and_test(int i,atomic_t *v)从v减i,如果结果为0就返回真,否则返回为假
int atomic_dec_and_test(int i,atomic_t *v)从v减1,如果结果为0就返回真,否则返回为假
int atomic_add_and_test(int i,atomic_t *v)从v加i,如果结果为0就返回真,否则返回为假
int atomic_inc_and_test(int i,atomic_t *v)从v加1,如果结果为0就返回真,否则返回为假

3.原子位操作API函数

原子位操作API函数
函数描述
void set_bit(int nr,void *p)将p地址的第nr位置1
void clear_bit(int nr,void *p)将p地址的第nr位清零
void change_bit(int nr,void *p)将p地址的第nr位翻转
int test_bit(int nr,void *p)获取p地址的第nr位的值
int test_and_set(int nr,void *p)将p地址的nr位置1,并返回nr位原来的值
int test_and_clear(int nr,void *p)将p地址的nr位清零,并返回nr位原来的值
int  test_and_change(int nr,void *p)将p地址的第nr位翻转,并返回nr位原来的值

三、自旋锁

1.简介

        当一个线程要访问某个共享资源的时候首先要先获取相应的锁,锁只能被一个线程持有,只要此线程不释放拥有的锁,那么其他的线程就不能获取此锁。

        自旋---原地打转

        缺点:时间短。

        使用结构体spinlock_t表示自旋锁。

2.自旋锁API函数

自旋锁API函数
函数描述
DEFINE_SPINLOCK(spinlock_t lock)定义并初始化一个自选变量
int spin_lock_init(spinlock_t *lock)初始化自旋锁

void spin_lock(spinlock_t *lock)

获取指定的自旋锁,加锁
void spin_unlock(spinlock_t *lock)释放指定的自旋锁
int spin_trylock(spinlock_t *lock)尝试获取指定的自旋锁,如果没有获取就返回0
int spin_is_locked(spinlock_t *lock)

检查指定的自旋锁是否被获取,如果没有被获取就返回非0,否则返回0

        自旋锁适用于线程与线程之间,被自旋锁保护的临界区一定不能调用任何能引起睡眠和阻塞的API函数,否则会导致死锁的发生。

        自旋锁会自动禁止抢占。

        中断里面可以使用自旋锁,但在中断里面使用自旋锁的时候,在获取之前一定要先禁止本地中断。相应的API函数如下:

函数描述
void spin_lock_irqsave(spinlock_t *lock,unsigned long flags)保存中断状态,禁止本地中断,并获取自旋锁。
void spin_unlock_irqrestore(spinlock_t *lock,unsigned long flags)将中断状态恢复打以前状态,并且激活本地中断,释放自旋锁。

        下半部也会竞争共享资源,下半部使用自旋锁的API函数有:

函数       描述
void spin_lock_bh(spinlock_t *lock)关闭下半部,并获取自旋锁
void spin_unlock_bh(spinlock_t lock)打开下半部,并释放自旋锁

3.其他类型的锁

        实际应用中用的不多,多的是在Linux内核中使用。

1)、读写自旋锁

2)、顺序锁

4.使用注意事项

1)、持有时间不能太长

2)、自旋锁保护的临界区内部能调用任何可能导致线程休眠的API函数,否则可能会导致死锁。

3)、不能递归申请自旋锁

4)、多核SOC编写程序

四、信号量

1.简介

        信号量常用于对共享资源的访问。

        信号量可以使线程进入休眠状态。

        信号量的开销比自旋锁要大。

        信号量特点:

1)、适用于占用资源比较长的场所。

2)、不能用于中断中。

        通过信号量控制访问资源的线程数。

        不能用于互斥访问。

2.信号量API函数

        使用semaphore结构体表示信号量。相关的API函数如下:

函数描述
DEFINE_SEMAPHORE(name)定义一个信号量,并设置信号量的值为1
void sema_init(struct semaphore *sem,int val)初始化信号量sem,设置信号量的值为val.
void down(struct semaphore *sem)获取信号量,不能用在中断中使用
int down_trylock(struct semaphore *sem)尝试获取信号量,能获取就返回0.如果不能返回非0,并且不会进入休眠。
int down_interruptible(struct semaphore *sem)获取信号量,进入休眠以后是可以被信号打断的。
void up(struct semaphore *sem)释放信号量

五、互斥体

1.简介

        一次只有一个线程访问共享资源,不能递归申请。需要互斥访问的时候建议使用mutex.

        注意以下几点:

1)、不能在中断中使用。

2)、保护的临界区可以调用引起阻塞的API函数。

3)、必须由mutex的持有者释放mutex。mutex不能递归上锁和解锁。

2.互斥体API函数

        相关的API函数有:

函数描述
DEFINE_MUXTEX(name)定义并初始化一个mutex变量
void mutex_init(mutex *lock)初始化mutex
void mutex_lock(struct mutex *lock)获取mutex  上锁  获取不到就休眠
void mutex_unlock(struct mutex *lock)释放mutex  解锁
iint mutex_trylock(struct mutex *lock)尝试获取mutex 成功返回0 失败返回0
int mutex_is_locked(struct mutex *lock)判断mutex是否被获取 获取返回1 否则返回0
int mutex_lock_interruptible(struct mutex *lock)使用此函数获取信号量失败进入休眠以后可以被信号打断

六、总结

        本篇笔记主要学习了相关的概念及API函数,并没有相关的案例进行说明。案例将在下一篇笔记中给出。本篇笔记主要内容包括原子操作、自旋锁、信号量和互斥体。

这篇关于Linux学习第18天:Linux并发与竞争: 没有规矩不成方圆的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/333629

相关文章

Linux搭建单机MySQL8.0.26版本的操作方法

《Linux搭建单机MySQL8.0.26版本的操作方法》:本文主要介绍Linux搭建单机MySQL8.0.26版本的操作方法,本文通过图文并茂的形式给大家讲解的非常详细,感兴趣的朋友一起看看吧... 目录概述环境信息数据库服务安装步骤下载前置依赖服务下载方式一:进入官网下载,并上传到宿主机中,适合离线环境

windows和Linux使用命令行计算文件的MD5值

《windows和Linux使用命令行计算文件的MD5值》在Windows和Linux系统中,您可以使用命令行(终端或命令提示符)来计算文件的MD5值,文章介绍了在Windows和Linux/macO... 目录在Windows上:在linux或MACOS上:总结在Windows上:可以使用certuti

Java并发编程之如何优雅关闭钩子Shutdown Hook

《Java并发编程之如何优雅关闭钩子ShutdownHook》这篇文章主要为大家详细介绍了Java如何实现优雅关闭钩子ShutdownHook,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 目录关闭钩子简介关闭钩子应用场景数据库连接实战演示使用关闭钩子的注意事项开源框架中的关闭钩子机制1.

Linux之systemV共享内存方式

《Linux之systemV共享内存方式》:本文主要介绍Linux之systemV共享内存方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、工作原理二、系统调用接口1、申请共享内存(一)key的获取(二)共享内存的申请2、将共享内存段连接到进程地址空间3、将

快速修复一个Panic的Linux内核的技巧

《快速修复一个Panic的Linux内核的技巧》Linux系统中运行了不当的mkinitcpio操作导致内核文件不能正常工作,重启的时候,内核启动中止于Panic状态,该怎么解决这个问题呢?下面我们就... 感谢China编程(www.chinasem.cn)网友 鸢一雨音 的投稿写这篇文章是有原因的。为了配置完

Java学习手册之Filter和Listener使用方法

《Java学习手册之Filter和Listener使用方法》:本文主要介绍Java学习手册之Filter和Listener使用方法的相关资料,Filter是一种拦截器,可以在请求到达Servl... 目录一、Filter(过滤器)1. Filter 的工作原理2. Filter 的配置与使用二、Listen

Linux命令之firewalld的用法

《Linux命令之firewalld的用法》:本文主要介绍Linux命令之firewalld的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux命令之firewalld1、程序包2、启动firewalld3、配置文件4、firewalld规则定义的九大

Linux之计划任务和调度命令at/cron详解

《Linux之计划任务和调度命令at/cron详解》:本文主要介绍Linux之计划任务和调度命令at/cron的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux计划任务和调度命令at/cron一、计划任务二、命令{at}介绍三、命令语法及功能 :at

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

Linux内核参数配置与验证详细指南

《Linux内核参数配置与验证详细指南》在Linux系统运维和性能优化中,内核参数(sysctl)的配置至关重要,本文主要来聊聊如何配置与验证这些Linux内核参数,希望对大家有一定的帮助... 目录1. 引言2. 内核参数的作用3. 如何设置内核参数3.1 临时设置(重启失效)3.2 永久设置(重启仍生效