0

0

【Linux】线程互斥之线程加锁

爱谁谁

爱谁谁

发布时间:2025-06-25 10:26:05

|

987人浏览过

|

来源于php中文网

原创

一、锁的定义

线程加锁是在多线程编程环境中,为了确保在同一时刻只有一个线程能够访问特定的共享资源或执行特定的代码段,而采取的一种同步手段,通过在需要保护的资源或代码段前获取锁,在访问完成后释放锁,来实现对共享资源的互斥访问

二、库函数1、初始化互斥锁代码语言:javascript代码运行次数:0运行复制
#include int pthread_mutex_init(pthread_mutex_t *restrict mutex, const pthread_mutexattr_t *restrict attr);

返回值:成功返回0,失败返回非零错误码 mutex:表示要初始化的互斥锁,pthread_mutex_t是POSIX线程库中定义的互斥锁类型 attr:包含互斥锁的属性,设置为NULL表示使用默认属性

2、销毁互斥锁代码语言:javascript代码运行次数:0运行复制
#include int pthread_mutex_destroy(pthread_mutex_t *mutex);

返回值:成功返回0,失败返回非零错误码 mutex:表示要销毁的互斥锁

3、加锁代码语言:javascript代码运行次数:0运行复制
#include int pthread_mutex_lock(pthread_mutex_t *mutex);

返回值:成功返回0,失败返回非零错误码 mutex:表示要加锁的互斥锁

4、解锁代码语言:javascript代码运行次数:0运行复制
#include int pthread_mutex_unlock(pthread_mutex_t *mutex);

返回值:成功返回0,失败返回非零错误码 mutex:表示要解锁的互斥锁

5、示例代码语言:javascript代码运行次数:0运行复制
#include #include #include #include #include using namespace std;//定义一个全局锁就可以不需要初始化和销毁锁的函数了//pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;#define NUM 4//共500张票int tickets = 500;class ThreadInfo{public:    ThreadInfo(const string &threadname, pthread_mutex_t *lock)    :threadname_(threadname)    ,lock_(lock)    {}public:    string threadname_;    pthread_mutex_t *lock_;};void *GrabTickets(void *args){    ThreadInfo *ti = static_cast(args);    string name(ti->threadname_);    while(true)    {        pthread_mutex_lock(ti->lock_); // 加锁        if(tickets > 0)        {            usleep(10000);            printf("%s get a ticket: %d\n", name.c_str(), tickets);            tickets--;            pthread_mutex_unlock(ti->lock_); // 解锁        }        else         {            pthread_mutex_unlock(ti->lock_); // 解锁            break;        }        //这里上面的代码        usleep(13); // 用休眠来模拟抢到票的后续动作    }    printf("%s quit...\n", name.c_str());}int main(){    pthread_mutex_t lock; // 定义互斥锁    pthread_mutex_init(&lock, nullptr); // 初始化互斥锁    vector tids;    vector tis;    for(int i = 1; i <= NUM; i++)    {        pthread_t tid;        ThreadInfo *ti = new ThreadInfo("Thread-"+to_string(i), &lock);        pthread_create(&tid, nullptr, GrabTickets, ti);        tids.push_back(tid);        tis.push_back(ti);    }    // 等待所有线程    for(auto tid : tids)    {        pthread_join(tid, nullptr);    }    // 释放资源    for(auto ti : tis)    {        delete ti;    }// 销毁互斥锁    pthread_mutex_destroy(&lock);         return 0;}
【Linux】线程互斥之线程加锁

这样就不会出现好多线程抢到一张票或者抢到不存在的票的问题了

三、深入理解锁1、解读锁的机制(一)先入为主原则

我们将上方代码中表示抢到票后续动作的休眠代码注释掉再次执行程序我们会发现,都是线程1抢的票,多次执行代码之后发现这是概率性问题,但是在抢票的时候,有一段时间的票都是一个线程抢到的,我们预想的应该是几乎平均分配的样子

【Linux】线程互斥之线程加锁

这说明了几个问题: 第一,线程对于锁的竞争能力不同,一定有一个首先抢到锁的线程 第二,一般来说,刚解锁再去抢锁的更容易一些,类似于上面的结果,一直是线程1在抢票

(二)锁和线程

对于上面第二个问题来说,我们有处理方法,这种方法就是同步,同步可以让所有的线程按照一定的顺序获取锁

对于其他线程来讲,一个线程要么获取到了锁,要么释放了锁,当前进程访问临界区的过程对于其他线程是原子的

有道智云AI开放平台
有道智云AI开放平台

有道智云AI开放平台

下载

在加锁期间,即解锁之前,是可以发生线程切换的,线程切换的时候是拿着锁走的,被锁起来的内容其他线程也是访问不到临界区的的,在该线程再次切换回来的时候,恢复线程上下文继续访问临界区代码

(三)锁的特点

加锁的本质就是用时间来换取安全,我们知道在加锁后,临界区的代码只能由一个线程执行,如果是并发执行,至少时间要缩短5倍,但是锁给我们消除了安全隐患,即可能出现的++--的隐患

加锁的表现就是线程对于临界区代码串行执行,一条线从上到下

我们加锁的原则就是尽量保证临界区的代码要少一些,可以使单线程执行的代码量更小,多线程综合处理的代码量更大,提高效率

锁的本身是共享资源,所以加锁和解锁本身就被设计成为了原子性操作(加锁和解锁通过硬件提供的原子指令,结合操作系统内核态的底层同步原语支持以及库层面的合理封装,来确保操作的原子性),这样可以确保在多线程环境下对共享资源加锁和解锁操作的完整性与一致性,避免因多线程并发干扰导致锁状态异常,进而保障线程安全和数据的正确性

2、锁的原理

下面来看一下加锁解锁对应的汇编指令,我们说,一条汇编指令就是原子性的

【Linux】线程互斥之线程加锁

首先al寄存器中的数字为0时,代表锁已被拿走,为非零(一般为1)时,代表锁当前空闲,可以上锁

加锁机制: movb $0, %al:将值 0 移动到 AL 寄存器xchgb %al, mutex:这是一个原子交换指令,将 AL 寄存器中的值(即 0)与 mutex 变量的值交换if (al寄存器的内容 > 0):检查 AL 寄存器中的内容(此时它保存的是原来 mutex 的值),如果值大于 0,说明互斥锁之前没有被锁定,锁定成功,返回 0else:如果 AL 中的值是 0,说明互斥锁已经被锁定,程序会等待goto lock:程序跳转回 lock 标签,重新尝试获取锁解锁机制: movb $1, mutex:将值 1 移动到 mutexxchgb %al, mutex:通过交换 AL 中的值和 mutex,实现解锁return 0:解锁后,函数返回四、锁的封装1、LockGuard.hpp代码语言:javascript代码运行次数:0运行复制
#pragma once#include //简单的封装了一下函数,用的时候方便一些class Mutex{public:    Mutex(pthread_mutex_t *lock)    :lock_(lock)    {}    void Lock()    {        pthread_mutex_lock(lock_);    }    void Unlock()    {        pthread_mutex_unlock(lock_);    }private:    pthread_mutex_t *lock_;};class LockGuard{public:    LockGuard(pthread_mutex_t *lock)    :mutex_(lock)    {        mutex_.Lock(); // 对象创建的时候加锁    }    ~LockGuard()    {        mutex_.Unlock(); // 对象销毁的时候解锁    }private:    Mutex mutex_;};
代码语言:javascript代码运行次数:0运行复制
#include #include #include #include #include #include "LockGuard.hpp"using namespace std;#define NUM 4int tickets = 500; //全局变量定义锁pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;class ThreadInfo{public:    ThreadInfo(const string &threadname)        : threadname_(threadname)public:    string threadname_;};void *GrabTickets(void *args){    ThreadInfo *ti = static_cast(args);    string name(ti->threadname_);    while (true)    {        {            LockGuard lockguard(&lock); // RAII 风格的锁            if (tickets > 0)            {                usleep(10000);                printf("%s get a ticket: %d\n", name.c_str(), tickets);                tickets--;            }            else            {                break;            }        }        usleep(13); // 用休眠来模拟抢到票的后续动作    }    printf("%s quit...\n", name.c_str());}int main(){    vector tids;    vector tis;    for (int i = 1; i <= NUM; i++)    {        pthread_t tid;        ThreadInfo *ti = new ThreadInfo("Thread-" + to_string(i));        pthread_create(&tid, nullptr, GrabTickets, ti);        tids.push_back(tid);        tis.push_back(ti);    }    // 等待所有线程    for (auto tid : tids)    {        pthread_join(tid, nullptr);    }    // 释放资源    for (auto ti : tis)    {        delete ti;    }    pthread_mutex_destroy(&lock);    return 0;}

这里封装的锁是RAII风格的锁,RAII风格是一种在 C++ 等编程语言中利用对象的构造和析构函数来自动管理资源的技术,确保资源在对象创建时获取,在对象生命周期结束时自动释放,以防止资源泄漏并简化资源管理

今日分享就到这里了~

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

556

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

374

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

733

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

477

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

414

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

1011

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

658

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

553

2023.09.20

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

52

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 6.3万人学习

Rust 教程
Rust 教程

共28课时 | 4.6万人学习

Excel 教程
Excel 教程

共162课时 | 12.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号