linux信号量及其实例

概述

Linux信号量是用于进程间同步和互斥的一种通信机制。本质是计数器
它们通常用于控制对共享资源的访问,以确保只有一个进程可以同时访问该资源。以下是一个详细的教程和C语言代码示例,展示如何使用信号量进行进程间通信。

创建信号量

要使用信号量,我们需要创建一个信号量集。每个信号量集由一个键(key)和一个初始值组成。我们可以使用 semget() 函数创建一个信号量集:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <stdio.h>

#define KEY 1234

int main()
{
    int semid = semget(KEY, 1, IPC_CREAT | 0666);
    if (semid < 0) {
        perror("semget error");
        return 1;
    }
    printf("Semaphore created: %d\n", semid);
    return 0;
}

在上面的代码中,我们使用 KEY 定义一个键值,以便多个进程可以访问同一个信号量集。然后我们使用 semget() 函数创建一个信号量集,该信号量集包含一个信号量,初始值为0。如果创建成功,该函数将返回信号量集的标识符,否则返回-1。

操作信号量

要对信号量进行操作,我们可以使用 semop() 函数。该函数允许我们执行3种操作:增加信号量的值、减少信号量的值、和等待信号量变为0。每个操作由一个结构体来表示,其中包含了信号量的标识符、要执行的操作和操作的参数。例如,以下代码将增加信号量的值:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <stdio.h>

#define KEY 1234

union semun {
    int val;
    struct semid_ds *buf;
    ushort *array;
};

int main()
{
    int semid = semget(KEY, 1, IPC_CREAT | 0666);
    if (semid < 0) {
        perror("semget error");
        return 1;
    }

    union semun arg;
    arg.val = 1;
    if (semctl(semid, 0, SETVAL, arg) < 0) {
        perror("semctl error");
        return 1;
    }

    struct sembuf buf;
    buf.sem_num = 0;
    buf.sem_op = 1;
    buf.sem_flg = SEM_UNDO;

    if (semop(semid, &buf, 1) < 0) {
        perror("semop error");
        return 1;
    }

    printf("Semaphore value: %d\n", semctl(semid, 0, GETVAL, arg));
    return 0;
}

在上面的代码中,我们使用 semctl() 函数将信号量的初始值设置为1。然后我们定义一个 sembuf 结构体,该结构体表示要对信号量进行的操作。在这种情况下,我们使用 semop() 函数执行一个操作,该操作将信号量的值增加1。最后,我们使用 semctl() 函数获取信号量的当前值,并将其打印出来。

除了增加信号量的值外,我们还可以使用 semop() 函数减少信号量的值。例如,以下代码将减少信号量的值:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <stdio.h>

#define KEY 1234

union semun {
    int val;
    struct semid_ds *buf;
    ushort *array;
};

int main()
{
    int semid = semget(KEY, 1, IPC_CREAT | 0666);
    if (semid < 0) {
        perror("semget error");
        return 1;
    }

    union semun arg;
    arg.val = 1;
    if (semctl(semid, 0, SETVAL, arg) < 0) {
        perror("semctl error");
        return 1;
    }

    struct sembuf buf;
    buf.sem_num = 0;
    buf.sem_op = -1;
    buf.sem_flg = SEM_UNDO;

    if (semop(semid, &buf, 1) < 0) {
        perror("semop error");
        return 1;
    }

    printf("Semaphore value: %d\n", semctl(semid, 0, GETVAL, arg));
    return 0;
}

在上面的代码中,我们使用 semop() 函数将信号量的值减少1。操作由 sembuf 结构体表示,其中 sem_op 的值为-1。最后,我们使用 semctl() 函数获取信号量的当前值,并将其打印出来。

销毁信号量

当不再需要使用信号量时,我们可以使用 semctl() 函数将其销毁。以下是一个示例代码:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <stdio.h>

#define KEY 1234

union semun {
    int val;
    struct semid_ds *buf;
    ushort *array;
};

int main()
{
    int semid = semget(KEY, 1, IPC_CREAT | 0666);
    if (semid < 0) {
        perror("semget error");
        return 1;
    }

    if (semctl(semid, 0, IPC_RMID, 0) < 0) {
        perror("semctl error");
        return 1;
    }

    printf("Semaphore destroyed\n");
    return 0;
}

在上面的代码中,我们使用 semctl() 函数将信号量集标识符为 semid 的信号量集销毁。操作由 IPC_RMID 表示。最后,我们打印出一条消息表示信号量已被销毁。

这是一个简单的Linux信号量通信的教程和C语言代码示例。注意,实际应用中可能需要使用更多的信号量以及更复杂的通信模式。

猜你喜欢

转载自blog.csdn.net/qq_46017342/article/details/129905336
今日推荐