【Linux】自旋锁

一、自旋锁概念

大家都知道互斥锁了,互斥锁呢就是如果这把锁被申请走了,那么该线程就阻塞在这里,直到这把锁被归还回来,那么此时线程拿到这把锁继续执行代码;

而自旋锁呢就是,锁被别的线程申请走了,不阻塞在这里,而是不断的轮询检查该锁是否被归还了;那么这个自旋锁的好处是:如果申请这把锁的线程很快的就把锁归还,那么就可以节约线程切换的开销(如果是阻塞的话:因为阻塞就是线程切换花费是时间还是很多的),但是如果规划的锁的时间非常长,自旋锁不断的轮询访问锁是否被归还,那么是非常浪费CPU资源的;

二、原理

自旋锁通常是使用一个标志位(例如布尔值)来表示该锁是否被占用;如果为 false 表示该锁没有被占用,那么线程就会把这这个布尔值修改成 true ,进入临界资源;如果为 true,表示该锁被其他线程占用了,线程就会循环访问该布尔值是否为 false,直到锁被归还;

伪代码讲解原理:

cpp 复制代码
// 自旋锁的本质:就是一个状态标记
// 0 = 锁是空闲的,可以获取
// 1 = 锁已经被占用
int lock = 0;

// 自旋锁加锁操作
void lock_spin() {
    // 死循环,一直重试(自旋核心)
    while (1) {
        // 原子交换:原子地把 lock 设置成 1,并返回旧值
        // 原子操作 = CPU 保证同一时刻只有一个线程能成功
        if (atomic_swap(&lock, 1) == 0) {
            // 旧值是 0 → 加锁成功!
            break;
        }
        // 加锁失败:不睡眠、不阻塞,立刻循环重试
        // 这就是"自旋":原地循环等待
    }
}

// 自旋锁解锁操作
void unlock_spin() {
    // 直接把标记恢复为 0 即可
    lock = 0;
}

三、优缺点分析

优点:

1)低延迟:自旋锁适用于短时间内的锁竞争情况,因为它不会让线程进入休眠状态,从而避免了线程切换的开销,提高了锁操作的效率。

2)减少系统调度开销:等待锁的线程不会被阻塞,不需要上下文切换,从而减少了系统调度的开销。

缺点:

1)CPU资源浪费:如果锁的持有时间较长,等待获取锁的线程会一直循环等待,导致CPU资源的浪费。

2)可能引起活锁:当多个线程同时自旋等待同一个锁时,如果没有适当的退避策略,可能会导致所有线程都在不断检查锁状态而无法进入临界区,形成活锁。我使用豆包来解析这段话:

注意:退避策略可以是休眠 usleep 1秒

四、使用场景

短暂等待的情况:适用于锁被占用时间很短的场景,如多线程对共享数据进行简单的读写操作。

多线程锁使用:通常用于系统底层,同步多个CPU对共享资源的访问。

注意:在使用自旋锁时,需要确保锁被释放的时间尽可能短,以避免CPU资源的浪费。在多CPU环境下,自旋锁可能不如其他锁机制高效,因为它可能导致线程在不同的CPU上自旋等待。使用豆包来解析这段话:

相关推荐
zh路西法1 天前
【navigation2全局路径更新频率修正】行为树框架的巧妙利用
linux
xieliyu.1 天前
Java算法精讲:双指针(二)
java·开发语言·算法
苏宸啊1 天前
IPC管道
linux·c++
bush41 天前
嵌入式linux学习记录十,定时器
linux·嵌入式
素材积累1 天前
博士后出站来深可申请的项目补贴等
数据库
何以解忧,唯有..1 天前
Python包管理工具pip:从入门到精通
开发语言·python·pip
BestOrNothing_20151 天前
ROS2 话题通信实战:消息对象、Publisher 发布器与 Subscriber 订阅器保姆级教程
c++·ros2·subscriber·publisher·话题通信
峥无1 天前
Linux进程信号:从基础概念到内核底层原理
linux·运维·服务器·信号处理
广州灵眸科技有限公司1 天前
瑞芯微RV1126B开发板(EASY-EAI-PI2) 开发(编译)方式说明
linux·服务器·单片机·嵌入式硬件·电脑
雪的季节1 天前
RabbitMQ详解
开发语言