Linux编程之信号量

在Linux编程中,通过fork调用创建子进程从而实现多进程非常容易,应用也非常广泛。多进程之间要保护临界区资源,方法之一就是使用信号量。Linux中的信号量API有两套,一套是POSIX标准的API,另一套则是较老的System V API。POSIX API遵循POSIX标准,因此移植性更好,即便是非UNIX/Linux系统,只要支持POSIX标准,代码也可编译运行;而System V API存在时间更长,因此可能存在的更广泛。下面分别介绍这两种API。

POSIX信号量

POSIX信号量有两种形式:

  • 命名信号量。命名信号量又一个名字标识,名字的形式为"/somename",即一个斜杠再加一些字母(只能又一个斜杠)。命名信号量可以用sem_open来创建或者打开,可以被多个进程同时操作。
  • 匿名信号量。匿名信号量则没有名字标识,根据使用范围,分为线程间共享的匿名信号量和进程间共享的匿名信号量。进程间共享的匿名信号量通常放在全局变量中;而进程间共享的信号量则放在共享内存中。匿名信号量使用sem_init函数进行初始化。

POSIX信号量相关的API如下。

       #include <fcntl.h>           /* For O_* constants */
       #include <sys/stat.h>        /* For mode constants */
       #include <semaphore.h>

       sem_t *sem_open(const char *name, int oflag);
       sem_t *sem_open(const char *name, int oflag,
                       mode_t mode, unsigned int value);

       int sem_init(sem_t *sem, int pshared, unsigned int value);

       int sem_wait(sem_t *sem);

       int sem_trywait(sem_t *sem);

       int sem_timedwait(sem_t *sem, const struct timespec *abs_timeout);

       int sem_post(sem_t *sem);

       int sem_close(sem_t *sem);

       int sem_unlink(const char *name);

       int sem_destroy(sem_t *sem);

使用POSIX信号量的基本步骤为:

  • 对于命名信号量,用sem_open创建或者打开,并指定名字。对于匿名信号量,应先申请sem_t存储,如果是进程间共享,可放到共享内存中;如果是线程间共享,则可放到全局变量中,然后调用sem_init函数初始化信号量。
  • 调用sem_wait等待信号量。
  • 访问临界区资源。
  • 调用sem_post释放信号量。
  • 对于命名信号量,调用sem_close释放信号量,并可调用sem_unlink删除。对于匿名信号量,调用sem_destroy删除信号量。

下面这个例子演示了命名信号量的使用。这个例子非常简单,父进程创建了两个子进程,然后三个进程都尝试获取信号量。从运行结果不难看出,由于初始值为1,因此刚开始两个进程尝试获取信号量只有1个成功,另一个等待;第一个获取信号量的进程释放后,后续进程才能再获取信号量。

/*
 * posix_sem.c: demostrates POSIX named semaphore
 */
#include <sys/stat.h>
#include <sys/wait.h>
#include <unistd.h>
#include <fcntl.h>
#include <semaphore.h>
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>

#define SEM_NAME "/test_posix_shm"
static sem_
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 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
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值