【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上自旋等待。使用豆包来解析这段话:

相关推荐
weelinking13 小时前
【2026】08_Claude与版本控制:Git协作技巧
数据库·人工智能·git·python·数据挖掘·交互·cloudera
sulikey17 小时前
Linux ext2文件系统结构
linux·操作系统·文件系统·linux文件系统·ext2·ext2文件系统
黄焖鸡能干四碗18 小时前
固定资产管理系统建设方案和源码(Java源码)
大数据·数据库·人工智能·物联网·区块链
白菜欣18 小时前
Linux — 进程控制
android·linux·运维
踩着两条虫18 小时前
「AI + 低代码」的可视化设计器
开发语言·前端·低代码·设计模式·架构
JoneBB18 小时前
ABAP Webservice连接
运维·开发语言·数据库·学习
解决问题no解决代码问题19 小时前
从乱码到脱敏导出:TiDB CSV 导出实战全指南
数据库
皮卡狮19 小时前
Linux开发专属工具
linux
未若君雅裁19 小时前
MySQL高可用与扩展-主从复制读写分离分库分表
java·数据库·mysql
2401_8676239819 小时前
CSS Flex布局中如何设置子元素间距_掌握gap属性的现代用法
jvm·数据库·python