linux 信号量 互斥量

一、什么是临界资源?
    (1)临界资源是一次仅允许一个任务使用的共享资源。每个任务(或中断处理handler)中访问临界资源的那段程序称为临界区。
    (2)在多任务系统中,为保障数据的可靠性和完整性,共享资源要互斥(独占)访问,所以全局变量(只读的除外)不能同时
       有多个任务访问,即一个任务访问的时候不能被其他任务打断。


二、如何保护临界资源?
信号量与互斥量(锁)
1、概念和定义
    信号量(Semaphore):
        多线程同步使用的
    互斥量(Mutex):
        多线程互斥使用的


2、信号量与互斥量的区别
    a、互斥量用于线程的互斥,信号量用于线程的同步:
        互斥:指某一资源同时只允许一个访问者对其进行访问,具有唯一性和排他性。但是互斥无法限制访问者对资源的访问
              顺序,所以访问是无序的;
        同步:指在互斥的基础上(多数情况),通过其他机制实现访问者对资源的有序访问。大多数情况下,同步已经实现了
              互斥,特别是所有写入资源的情况必定是互斥的。少数情况指可以允许多个访问者同时访问资源。

    b、互斥量值只能是0/1,信号量值可以为非负整数:
        一个互斥量只能用于一个资源的多线程互斥访问,不能实现多个资源的多线程互斥问题;
        一个信号量可以实现多个同类资源的多线程互斥和同步。当信号量为单值信号量时,也可以完成一个资源的互斥访问;

    c、互斥量的加锁和解锁必须由同一线程分别对应使用
       而信号量可以由一个线程释放,另外一个线程得到。
    

3、信号量与互斥量的详细分析
    信号量--Semaphore:
        分类:
            二进制信号量(Binary Semaphore):只允许信号量取0或1值,其同时只能被一个线程获取;
            整型信号量(Integer Semaphore):信号量取值是整数,可被多个线程同时获得,直到信号量的值变为0;
            记录型信号量(Record Semaphore):每个信号量除了有一个整数值value(计数)外,还有一个等待队列List,其中是阻
                                              塞在该信号量的各个线程的标识。当信号量被释放1个,值被加1后,系统自动从等待
                                              队列中唤醒一个等待中的线程;让其获得信号量,同时信号量再减1。
        Semaphore 抽象操作:
            create() / wait() / post() / trywait() / destroy()
    
    互斥量--Mutex:
        互斥量表现互斥现象的数据结构,常用作保护从中断来的临界段代码对资源的访问;
        Mutex本质就是一把锁,提供对资源的独占式访问,所以Mutex主要的作用是互斥。
        Mutex 抽象操作:
            create() / lock() / unlock() / destroy()
            

三、引申一下
    锁可以作如下分类:
        睡眠锁:
            信号量
            互斥量
        非睡眠锁:
            spinlock
            

            

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
互斥信号量Linux内核中一种用于实现进程或线程之间互斥访问共享资源的机制。它可以确保在同一时间只有一个进程或线程可以访问共享资源,从而避免了竞争条件和数据不一致的问题。 在Linux内核中,互斥信号量通过信号量机制来实现。当一个进程或线程需要访问共享资源时,它可以调用down()函数来获取互斥信号量。如果互斥信号量当前不可用(即已被其他进程或线程获取),调用down()函数将使该进程或线程进入睡眠状态,直到互斥信号量被释放。 下面是一个示例代码,演示了如何使用互斥信号量来实现进程之间的互斥访问: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #include <sys/ipc.h> #include <sys/sem.h> #define KEY 1234 int main() { int semid; struct sembuf sb; // 创建一个互斥信号量 semid = semget(KEY, 1, IPC_CREAT | 0666); if (semid == -1) { perror("semget"); exit(1); } // 初始化互斥信号量的值为1 if (semctl(semid, 0, SETVAL, 1) == -1) { perror("semctl"); exit(1); } // 获取互斥信号量 sb.sem_num = 0; sb.sem_op = -1; sb.sem_flg = 0; if (semop(semid, &sb, 1) == -1) { perror("semop"); exit(1); } // 临界区代码 printf("进程1进入临界区\n"); sleep(5); printf("进程1离开临界区\n"); // 释放互斥信号量 sb.sem_op = 1; if (semop(semid, &sb, 1) == -1) { perror("semop"); exit(1); } // 删除互斥信号量 if (semctl(semid, 0, IPC_RMID) == -1) { perror("semctl"); exit(1); } return 0; } ``` 在上面的示例代码中,我们使用了System V信号量机制来创建和操作互斥信号量。首先,我们使用semget()函数创建一个互斥信号量,并使用semctl()函数将其初始化为1。然后,我们使用semop()函数调用down()操作来获取互斥信号量,进入临界区执行一些操作,然后再使用semop()函数调用up()操作来释放互斥信号量。 请注意,上述示例代码是使用C语言编写的,需要在Linux环境下编译和运行。另外,互斥信号量的具体实现可能因操作系统和编程语言而有所不同。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

denglin12315

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值