把信号灯集加到共享内存实现同步:输入输出,quit结束

目录

题目

思路

代码

input.c

output.c


题目

把信号灯集加到共享内存实现同步:输入输出,quit结束

思路

一、整体功能概述

 

这段代码使用共享内存和信号量集实现了两个进程之间的同步通信。一个进程负责从终端输入内容并存入共享内存,另一个进程从共享内存读取内容并输出,当输入为 “quit” 时,两个进程都结束。

 

二、主要函数和数据结构分析

 
  1. union semun

    • 用于设置信号量的值。它包含一个整数成员 val,可以用来初始化或修改信号量的值。
  2. init 函数:

    • 接受信号量集标识符 semid、信号量编号 num 和初始值 val 作为参数。
    • 创建一个 union semun 类型的变量 sem,并将传入的 val 赋值给 sem.val
    • 使用 semctl 函数将指定编号的信号量初始化为传入的 val
  3. pv 函数:

    • 接受信号量集标识符 semid、信号量编号 num 和操作值 op 作为参数。
    • 创建一个 struct sembuf 类型的变量 buf,设置信号量编号、操作值和标志位(这里设置为阻塞)。
    • 使用 semop 函数对指定编号的信号量执行操作。
  4. input 进程(主函数):

    • 生成共享内存和信号量集的键值 key,使用 ftok 函数以指定的文件("sem.c")和字符('a')生成唯一键值。
    • 创建或打开共享内存,通过 shmget 和 shmat 函数获取共享内存的标识符和映射地址。
    • 创建或打开信号量集,通过 semget 函数获取信号量集标识符。如果信号量集已存在,则初始化其中的两个信号量为 0 和 1。
    • 在一个无限循环中,先对 1 号信号量执行 P 操作(pv(semid, 1, -1)),等待资源可用以进行输入操作。从终端读取输入并存入共享内存,然后对 0 号信号量执行 V 操作(pv(semid, 0, 1)),通知输出进程可以读取共享内存中的内容。如果输入为 “quit”,则跳出循环。
    • 最后取消共享内存的映射,但没有删除共享内存和信号量集(注释掉了相关代码)。
  5. output 进程(主函数):

    • 与 input 进程类似,生成键值、创建或打开共享内存和信号量集。
    • 在一个无限循环中,先对 0 号信号量执行 P 操作(pv(semid, 0, -1)),等待输入进程将内容存入共享内存。如果读取到的内容为 “quit”,则跳出循环。否则,输出共享内存中的内容,并对 1 号信号量执行 V 操作(pv(semid, 1, 1)),通知输入进程可以进行下一次输入操作。
    • 最后取消共享内存的映射,删除共享内存和信号量集。
 

三、执行流程分析

 
  1. 两个进程分别执行各自的 main 函数。
  2. 首先生成相同的键值,确保能够访问到相同的共享内存和信号量集。
  3. input 进程创建或打开共享内存和信号量集后,等待 1 号信号量资源可用,然后从终端读取输入并存入共享内存,接着释放 0 号信号量通知 output 进程。
  4. output 进程等待 0 号信号量资源可用,然后读取共享内存中的内容并输出,接着释放 1 号信号量通知 input 进程。
  5. 如此循环,直到 input 进程输入 “quit”,两个进程都检测到这个字符串并跳出循环,结束执行。
 

四、应用场景和注意事项

 
  1. 应用场景:

    • 适用于需要两个或多个进程之间进行同步通信的场景,特别是当需要在进程之间共享数据并确保数据的一致性和顺序性时。
    • 可以用于实现生产者 - 消费者模型,其中一个进程作为生产者将数据存入共享内存,另一个进程作为消费者从共享内存读取数据。
  2. 注意事项:

    • 确保在使用共享内存和信号量集时进行适当的错误处理,以避免程序出现异常情况。
    • 考虑在程序结束时正确地删除共享内存和信号量集,以释放系统资源。
    • 注意信号量的初始值设置和操作顺序,以确保进程之间的同步正确无误。

代码

input.c

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

union semun
{
    int val;
};

//初始化
void init(int semid, int num, int val)
{
    union semun sem;
    sem.val = val;
    semctl(semid, num, SETVAL, sem); //把num号信号灯初始化值为val
}

//PV操作
void pv(int semid, int num, int op)
{
    struct sembuf buf; //sembuf结构体人家写好的直接拿来用就可以
    buf.sem_num = num;
    buf.sem_op = op;
    buf.sem_flg = 0;       //阻塞
    semop(semid, &buf, 1); //对num号灯进行op操作
}

int main(int argc, char const *argv[])
{
    key_t key;
    int semid;
    int shmid;
    char *p;

    if ((key = ftok("sem.c", 'a')) < 0)
    {
        perror("ftok err");
        return -1;
    }
    printf("key: %#x\n", key);

    //创建或打开共享内存
    shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0777);
    if (shmid <= 0)
    {
        if (errno == EEXIST)
            shmid = shmget(key, 128, 0777);
        else
        {
            perror("shmget er");
            return -1;
        }
    }
    printf("shmid: %d\n", shmid);

    //映射共享内存
    p = (char *)shmat(shmid, NULL, 0);
    if (p == (char *)-1)
    {
        perror("shmat err");
        return -1;
    }

    //创建或打开信号灯集
    semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0777);
    if (semid <= 0)
    {
        if (errno == EEXIST)
            semid = semget(key, 2, 0777);
        else
        {
            perror("semget er");
            return -1;
        }
    }
    else //确保对信号灯集中的信号灯始终初始化一次,下次执行程序还继续获得上一次的值。如果修改初值,需要删除信号灯然后重新打开。
    {
        //初始化信号灯
        init(semid, 0, 0); //把0号信号灯初始化值为0
        init(semid, 1, 1); //把1号信号灯初始化值为1
    }
    printf("semid: %d\n", semid);

    //获取信号灯值
    printf("sem0: %d\n", semctl(semid, 0, GETVAL));
    printf("sem1: %d\n", semctl(semid, 1, GETVAL));

    while (1)
    {
        pv(semid, 1, -1);
        scanf("%s", p);
        pv(semid, 0, 1);
        if (strcmp(p, "quit") == 0)
            break;
    }

    shmdt(p);                      //取消映射
    // shmctl(shmid, IPC_RMID, NULL); //删除共享内存
    // semctl(semid, 0, IPC_RMID);    //删除信号灯集
    return 0;
}

output.c

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

union semun
{
    int val;
};

//初始化
void init(int semid, int num, int val)
{
    union semun sem;
    sem.val = val;
    semctl(semid, num, SETVAL, sem); //把num号信号灯初始化值为val
}

//PV操作
void pv(int semid, int num, int op)
{
    struct sembuf buf; //sembuf结构体人家写好的直接拿来用就可以
    buf.sem_num = num;
    buf.sem_op = op;
    buf.sem_flg = 0;       //阻塞
    semop(semid, &buf, 1); //对num号灯进行op操作
}

int main(int argc, char const *argv[])
{
    key_t key;
    int semid;
    int shmid;
    char *p;

    if ((key = ftok("sem.c", 'a')) < 0)
    {
        perror("ftok err");
        return -1;
    }
    printf("key: %#x\n", key);

    //创建或打开共享内存
    shmid = shmget(key, 128, IPC_CREAT | IPC_EXCL | 0777);
    if (shmid <= 0)
    {
        if (errno == EEXIST)
            shmid = shmget(key, 128, 0777);
        else
        {
            perror("shmget er");
            return -1;
        }
    }
    printf("shmid: %d\n", shmid);

    //映射共享内存
    p = (char *)shmat(shmid, NULL, 0);
    if (p == (char *)-1)
    {
        perror("shmat err");
        return -1;
    }

    //创建或打开信号灯集
    semid = semget(key, 2, IPC_CREAT | IPC_EXCL | 0777);
    if (semid <= 0)
    {
        if (errno == EEXIST)
            semid = semget(key, 2, 0777);
        else
        {
            perror("semget er");
            return -1;
        }
    }
    else //确保对信号灯集中的信号灯始终初始化一次,下次执行程序还继续获得上一次的值。如果修改初值,需要删除信号灯然后重新打开。
    {
        //初始化信号灯
        init(semid, 0, 0); //把0号信号灯初始化值为0
        init(semid, 1, 1); //把1号信号灯初始化值为1
    }
    printf("semid: %d\n", semid);

    //获取信号灯值
    printf("sem0: %d\n", semctl(semid, 0, GETVAL));
    printf("sem1: %d\n", semctl(semid, 1, GETVAL));

    while (1)
    {
        pv(semid, 0, -1);
        if (strcmp(p, "quit") == 0)
            break;
        printf("shm: %s\n", p);
        pv(semid, 1, 1);
    }

    shmdt(p);                      //取消映射
    shmctl(shmid, IPC_RMID, NULL); //删除共享内存
    semctl(semid, 0, IPC_RMID);    //删除信号灯集
    return 0;
}

  • 7
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,让我来解答你的问题。 实现两个进程之间的通信,可以使用共享内存信号灯集。共享内存是一种让多个进程可以访问同一块内存的机制,而信号灯集则可以用来对共享内存进行同步。 以下是实现过程: 1. 创建共享内存信号灯集 首先,我们需要创建一个共享内存和一个信号灯集。我们可以使用shmget函数和semget函数来创建它们: ``` int shmid = shmget(IPC_PRIVATE, sizeof(char) * 1024, IPC_CREAT | 0666); char *shared_memory = (char *)shmat(shmid, NULL, 0); int semid = semget(IPC_PRIVATE, 1, IPC_CREAT | 0666); ``` 这里我们创建了一个大小为1024的字符型共享内存区域和一个信号灯集,其中semget函数的第二个参数表示信号灯集中的信号灯数量,这里只需要一个。 2. 初始化信号灯 接下来,我们需要对信号灯进行初始化。我们可以使用semctl函数来完成: ``` union semun { int val; struct semid_ds *buf; ushort *array; }; union semun sem_union; sem_union.val = 1; if (semctl(semid, 0, SETVAL, sem_union) == -1) { perror("Error initializing semaphore"); exit(EXIT_FAILURE); } ``` 这里我们将信号灯的值初始化为1,表示可以访问共享内存区域。 3. 创建子进程 接下来,我们需要创建两个子进程,一个用来从终端输入,一个用来打印共享内存区域中的内容。我们可以使用fork函数来创建子进程: ``` pid_t pid = fork(); if (pid == -1) { perror("Error forking process"); exit(EXIT_FAILURE); } else if (pid == 0) { // 子进程1,用来从终端输入 // ... } else { pid_t pid2 = fork(); if (pid2 == -1) { perror("Error forking process"); exit(EXIT_FAILURE); } else if (pid2 == 0) { // 子进程2,用来打印共享内存区域中的内容 // ... } else { // 父进程 // ... } } ``` 这里我们使用了两次fork函数来创建两个子进程,如果fork函数返回0,则表示是子进程,否则是父进程。 4. 子进程1读取终端输入并写入共享内存 在子进程1中,我们需要循环读取终端输入,并将输入的内容写入共享内存区域。我们可以使用fgets函数来读取终端输入: ``` while (1) { printf("Please input: "); fgets(shared_memory, 1024, stdin); sembuf sem_buffer = {0, -1, 0}; semop(semid, &sem_buffer, 1); if (strcmp(shared_memory, "quit\n") == 0) { break; } sem_buffer.sem_op = 1; semop(semid, &sem_buffer, 1); } ``` 这里我们使用了sembuf结构体和semop函数来对信号灯进行操作,当写入共享内存时,我们需要将信号灯的值减1,表示共享内存区域被占用,当写入完成后,我们需要将信号灯的值加1,表示共享内存区域可以被其他进程访问。 5. 子进程2循环读取共享内存并打印 在子进程2中,我们需要循环读取共享内存区域中的内容,并将其打印到终端上。我们可以使用printf函数来完成: ``` while (1) { sembuf sem_buffer = {0, -1, 0}; semop(semid, &sem_buffer, 1); printf("Received: %s", shared_memory); sem_buffer.sem_op = 1; semop(semid, &sem_buffer, 1); if (strcmp(shared_memory, "quit\n") == 0) { break; } } ``` 这里我们需要先将信号灯的值减1,表示我们要读取共享内存区域中的内容,读取完成后,我们需要将信号灯的值加1,表示共享内存区域可以被其他进程访问。 6. 父进程等待子进程结束并删除共享内存信号灯 在父进程中,我们需要等待两个子进程结束,并删除共享内存信号灯。我们可以使用wait函数来等待子进程结束: ``` waitpid(pid, NULL, 0); waitpid(pid2, NULL, 0); ``` 等待子进程结束后,我们可以使用shmctl函数和semctl函数来删除共享内存信号灯: ``` shmdt(shared_memory); shmctl(shmid, IPC_RMID, NULL); semctl(semid, 0, IPC_RMID, sem_union); ``` 这里我们使用了shmdt函数将共享内存从进程中分离,然后使用shmctl函数将其删除。对于信号灯集,我们使用semctl函数将其删除。 以上就是使用共享内存信号灯实现两个进程通信的过程。希望能对你有所帮助!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值