Linux信号量sem_init,sem_wait,sem_post

本文介绍了信号量在编程中的概念,特别是二进制信号量如何用于线程同步,以及四个基本的信号量接口函数sem_init,sem_wait,sem_post和sem_destroy的使用方法和示例。展示了如何在C语言中使用pthread库实现线程间的同步控制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、什么是信号量
线程的信号量与进程间通信中使用的信号量的概念是一样,它是一种特殊的变量,它可以被增加或减少,但对其的关键访问被保证是原子操作。如果一个程序中有多个线程试图改变一个信号量的值,系统将保证所有的操作都将依次进行。

而只有0和1两种取值的信号量叫做二进制信号量,在这里将重点介绍。而信号量一般常用于保护一段代码,使其每次只被一个执行线程运行。我们可以使用二进制信号量来完成这个工作。

二、信号量的接口函数
信号量的函数都以sem_开头,线程中使用的基本信号量函数有4个,它们都声明在头文件semaphore.h中。

sem_init函数
该函数用于创建信号量,其原型如下:

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

该函数初始化由sem指向的信号对象,设置它的共享选项,并给它一个初始的整数值。
pshared控制信号量的类型,如果其值为0,就表示这个信号量是当前进程的局部信号量,否则信号量就可以在多个进程之间共享,value为sem的初始值。调用成功时返回0,失败返回-1.

sem_wait函数
该函数用于以原子操作的方式将信号量的值减1。原子操作就是,如果两个线程企图同时给一个信号量加1或减1,它们之间不会互相干扰。它的原型如下:

int sem_wait(sem_t *sem);  

sem_post函数

该函数用于以原子操作的方式将信号量的值加1。它的原型如下:

int sem_post(sem_t *sem); 

与sem_wait一样,sem指向的对象是由sem_init调用初始化的信号量。调用成功时返回0,失败返回-1.

sem_destroy函数

该函数用于对用完的信号量的清理。它的原型如下:

int sem_destroy(sem_t *sem); 

成功时返回0,失败时返回-1.

三、信号量的函数使用

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>
#include <semaphore.h>

sem_t sem;
void* func1(void* arg)
{
    sem_wait(&sem);
    printf("thread1 running\n");
    return NULL;
}
void* func2(void* arg)
{
    printf("thread2 running\n");
    sem_post(&sem);
    return NULL;
}
int main()
{
    sem_init(&sem,0,0);
    pthread_t thread[2];
    pthread_create(&(thread[0]),NULL,func1,NULL);
    printf("main thread running\n");
    for (int i = 0; i < 5; ++i) {
        printf("i = %d\n",i);
        sleep(1);
    }
    pthread_create(&(thread[1]),NULL,func2,NULL);
    printf("main thread running2\n");
    pthread_join(thread[0],NULL);
    pthread_join(thread[1],NULL);
    sem_destroy(&sem);
    return 0;
}
`sem_init` 函数是 POSIX 信号量接口中的一个函数,用于创建一个未命名的(匿名的)POSIX 信号量,并进行初始化。该函数在 Linux 和其他符合 POSIX 标准的操作系统中都存在,它为进程间或线程间的同步提供了一种机制。 函数原型如下: ```c int sem_init(sem_t *sem, int pshared, unsigned int value); ``` 参数说明: 1. `sem`:指向 sem_t 类型的指针,该指针指向将要初始化的信号量对象。 2. `pshared`:指定信号量的类型,用于标识信号量将在哪些进程间共享。如果该值为 0,则信号量仅在调用 sem_init 的进程内共享;如果该值非 0,则信号量可以在多个进程间共享。 3. `value`:信号量的初始值。该值必须大于或等于 0,表示资源的可用数量。 函数执行成功时返回 0,失败时返回 -1,并设置相应的错误码。 需要注意的是,在多线程编程中,如果一个信号量被多个进程共享(`pshared` 非 0),那么在调用 `sem_init` 之后,所有需要访问该信号量的进程都必须调用 `sem_init`,并且传入相同的 `value` 值。而当信号量仅在同一进程内的多个线程之间共享时(`pshared` 为 0),信号量的初始化只需要在进程内的一个线程中完成即可。 使用 `sem_init` 创建并初始化信号量后,可以通过 `sem_wait`、`sem_post` 等函数来进行信号量的 P(等待)和 V(信号)操作。 在使用完信号量之后,应该调用 `sem_destroy` 函数来销毁信号量,释放相关资源。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值