簡體   English   中英

C-帶循環的多個子叉

[英]C - Multiple child fork with loops

因此,我必須使用fork創建7個子進程,並且每個子進程都必須循環工作,直到medicamentos(meds)用完為止。 但是它們一次只能運行一次,有時只能運行一次。 變量和其他名稱在pt-br中,但我將進行翻譯。 因此,它本來應該是療養院(糟糕透頂),而且葯物數量有限。 一個人需要服用1粒葯才能睡4個小時,需要8粒才能睡8個小時。 此外,只有5張床,因此需要等待這些床。 一旦有一張免費的床,一個醒着的等待長者的長者會隨機服用1或2片葯,然后入睡。 在那之后,他將離開床,讓另一個老人入睡。

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>

#define ACORDADO 0 //awake
#define ESPERANDO 1 //waiting
#define DORMINDO 2 //sleeping
#define N 7

void mostrar(void); //show info
void dormir(int); //sleep
void tomar_med(int); //take meds
void esperar(int);//wait
void acordar(int);//wake up
void liberar_cama(int);//leave the bed
void acao_idoso(int j);//elder action

int medicamentos = 90; //meds
int camas = 5; //beds
int idosos [N]; //elders
int i;

void forker(int n_proc)
{
    pid_t pid;

    if(n_proc > 0)
    {
        if ((pid = fork()) < 0){
            perror("fork");

        }else if (pid == 0){ //filho
            acao_idoso(n_proc - 1);

        }else if(pid > 0){ //pai
            mostrar();
            forker(n_proc - 1);
        }
    }
}

void main(){


    forker(7);

}

void mostrar(){
    for(i=0;i<N;i++){
        if(idosos[i] == ACORDADO)
            printf("O idoso %d esta acordado.\n", i+1); //elder is awake
        if(idosos[i] == ESPERANDO)
            printf("O idoso %d esta esperando.\n", i+1); //elder is waiting
        if(idosos[i] == DORMINDO)
            printf("O idoso %d esta dormindo.\n", i+1); //elder is asleep
    }
    printf("\n");   
    printf("%d medicamentos sobrando.\n", medicamentos); //%d meds left
    printf("\n");
}

void acao_idoso(int j){
    int i = j;
    while(medicamentos != 0){
        acordar(i);
        mostrar();
        esperar(i);
        mostrar();
        dormir(i);
        mostrar();
        liberar_cama(i);
        mostrar();
    }
}

void acordar(int i){
    if(idosos[i] == ACORDADO){
        int t = rand() %11;
        sleep(t+1);
    }
}

void esperar(int i){
    if(idosos[i] == ACORDADO && medicamentos > 0)
        idosos[i] = ESPERANDO;
}

void dormir(int i){
    if((idosos[i] == ESPERANDO) && (camas >= 1)){
        int j = rand();

        if(i % 2 == 0){
            medicamentos = medicamentos - 2;
            sleep(8);
        }else{
            medicamentos = medicamentos - 1;
            sleep(4);
        }

        camas = camas - 1;
        idosos[i] = DORMINDO;

    }
}

void liberar_cama(int i){
    if(idosos[i] == DORMINDO){
        idosos[i] = ACORDADO;
        camas++;
    }
}

如注釋中所述,您的主程序需要等待子dormir()完成,並且需要在dormir()使用j

如果您一次睡了4、8秒,則需要較長時間(10分鍾以上)才能完成。 改變睡眠時調用sleep10(int n)睡眠n十分之一秒,並提供實施使用nanosleep()在東西好像75秒代碼完成。

您還需要確保僅剩一粒葯時,沒有人可以服用2粒。 獲得負面的“ sobrando sobrando”使程序無限期地運行。

在Linux上, void main()是無條件錯誤的。 使用int main(void)和(最好)使用顯式return 0; main()的末尾,除非您有其他返回狀態。

該代碼還會在消息之前打印出PID號,以標識哪個進程在做什么。 有空間認為應該打印更多的信息,但是我的西班牙語已經不存在了,所以我不會嘗試。 main()程序報告子項及其退出狀態。

#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/wait.h>
#include <unistd.h>
#include <time.h>

#define ACORDADO 0
#define ESPERANDO 1
#define DORMINDO 2
#define N 7

void mostrar(void);
void dormir(int);
void tomar_med(int);
void esperar(int);
void acordar(int);
void liberar_cama(int);
void acao_idoso(int j);
void forker(int n_proc);

int medicamentos = 90;
int camas = 5;
int idosos[N];

int main(void)
{
    forker(N);
    int corpse;
    int status;
    while ((corpse = wait(&status)) > 0)
        printf("PID %5d exited with status 0x%.4X\n", corpse, status);
    return 0;
}

static void sleep10(int n)
{
    assert(n > 0 && n < 100);
    struct timespec ts = { .tv_sec = n / 10, .tv_nsec = (n % 10) * 100000000 };
    printf("%d: sleeping %d.%ds\n", (int)getpid(), n / 10, n % 10);
    nanosleep(&ts, 0);
}

void forker(int n_proc)
{
    pid_t pid;

    if (n_proc > 0)
    {
        if ((pid = fork()) < 0)
        {
            perror("fork");
        }
        else if (pid == 0)
        {
            printf("%d: PID %d\n", n_proc, (int)getpid());
            acao_idoso(n_proc - 1);
        }
        else if (pid > 0)
        {
            mostrar();
            forker(n_proc - 1);
        }
    }
}

void mostrar()
{
    int pid = getpid();
    for (int i = 0; i < N; i++)
    {
        if (idosos[i] == ACORDADO)
            printf("%d: O idoso %d esta acordado.\n", pid, i + 1);
        if (idosos[i] == ESPERANDO)
            printf("%d: O idoso %d esta esperando.\n", pid, i + 1);
        if (idosos[i] == DORMINDO)
            printf("%d: O idoso %d esta dormindo.\n", pid, i + 1);
    }
    printf("\n");
    printf("%d: %d medicamentos sobrando.\n", pid, medicamentos);
    printf("\n");
}

void acao_idoso(int j)
{
    int i = j;
    while (medicamentos != 0)
    {
        acordar(i);
        mostrar();
        esperar(i);
        mostrar();
        dormir(i);
        mostrar();
        liberar_cama(i);
        mostrar();
    }
}

void acordar(int i)
{
    if (idosos[i] == ACORDADO)
    {
        int t = rand() % 11;
        sleep10(t + 1);
    }
}

void esperar(int i)
{
    if (idosos[i] == ACORDADO && medicamentos > 0)
        idosos[i] = ESPERANDO;
}

static inline int min(int x, int y) { return (x < y) ? x : y; }

void dormir(int i)
{
    if ((idosos[i] == ESPERANDO) && (camas >= 1))
    {
        int j = rand();

        if (j % 2 == 0)
        {
            medicamentos = medicamentos - min(2, medicamentos);
            sleep10(8);
        }
        else
        {
            medicamentos = medicamentos - 1;
            sleep10(4);
        }

        camas = camas - 1;
        idosos[i] = DORMINDO;
    }
}

void liberar_cama(int i)
{
    if (idosos[i] == DORMINDO)
    {
        idosos[i] = ACORDADO;
        camas++;
    }
}

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM