c信号量和共享内存动作计数器

时间:2013-04-23 09:19:04

标签: c shared-memory semaphore

我正在使用信号量和共享内存。我需要创建11个子进程。首先,I = 0打印A,其他打印B.然后我需要每个进程打印带有计数器*APtr的消息,因此输出上的行将带有数字1-11,后跟进程指示和最终值* APtr在最后一行应该是12。

现在它不起作用了。进程正在同时写入和读取* APtr,因此我得到第一个数字的输出,例如1,1,1,4,5,6,7,7,8,8,11,12

int I, A;
int *APtr;
//create shared memory (action counter)
if((A=shmget(ftok(argv[0],getpid()), sizeof(int),  IPC_CREAT | 0660)) == -1)
{
    exit(2);
}
if((APtr=shmat(A,NULL,0)) == NULL)
{
    exit(2);
}
*APtr = 1;


/*semaphor semA*/
sem_t semA;
if(sem_init(&semA, 0, 1)==-1)
{
    exit(2);
}
/*semafor semA*/
pid_t pid;
for(int i=0; i<11; ++i)
{
    if((pid = fork()) < 0)
    {     //chyba spusteni noveho procesu
        kill(0, SIGTERM);
        exit(2);
    }
    else if(pid == 0)
    {
        I=i;
        if(I == 0)//A
        {
            sem_wait(&semA);
            //int pom = *APtr;
            fprintf(stdout, "%d: A %d:\n", *APtr, I);
            fflush(stdout);
            *APtr+=1;
            sem_post(&semA);
        }
        else//B
        {
            sem_wait(&semA);
            //int pom = *APtr;
            fprintf(stdout, "%d: B %d:\n", *APtr, I);
            fflush(stdout);
            *APtr+=1;
            sem_post(&semA);
        }
        break;
    }
    else
    {
        //fprintf(stdout, "%d\n", *APtr);
    }
}
sem_destroy(&semA);
if(pid != 0)
{
    sleep(1);
    fprintf(stdout, "%d\n", *APtr);
}
return 0;

如果我理解正确sem_init(&semA, 0, 1);应初始化信号量,以便在sem_wait(&semA);之后只允许一个过程工作,直到sem_post(&semA);命令完成。那么我在做什么/理解错误呢?


切换到sem_open()后,我返回NULL并且所有分叉进程在sem_wait()上崩溃,有我的新代码。

int I, A;
int *APtr;
//create shared memory (action counter)
if((A=shmget(ftok(argv[0],getpid()), sizeof(int),  IPC_CREAT | 0660)) == -1)
{
    exit(2);
}
if((APtr=shmat(A,NULL,0)) == NULL)
{
    exit(2);
}
*APtr = 1;


/*semaphor semA*/
sem_t *semA;
if(semA = sem_open("/semA", O_CREAT, 0700, 1) == SEM_FAILED)
{
    exit(2);
}
/*semafor semA*/
pid_t pid;
for(int i=0; i<11; ++i)
{
    if((pid = fork()) < 0)
    {     //chyba spusteni noveho procesu
        kill(0, SIGTERM);
        exit(2);
    }
    else if(pid == 0)
    {
        I=i;
        if(I == 0)//A
        {
            sem_wait(semA);
            fprintf(stdout, "%d: A %d:\n", *APtr, I);
            fflush(stdout);
            *APtr+=1;
            sem_post(semA);
        }
        else//B
        {
            sem_wait(semA);
            fprintf(stdout, "%d: B %d:\n", *APtr, I);
            fflush(stdout);
            *APtr+=1;
            sem_post(semA);
        }
        break;
    }
    else
    {
        //fprintf(stdout, "%d\n", *APtr);
    }
}
if(pid != 0)
{
    sleep(1);
    sem_close(semA);
    fprintf(stdout, "%d\n", *APtr);
}
return 0;

2 个答案:

答案 0 :(得分:1)

sem_t semA;
if(sem_init(&semA, 0, 1)==-1)

这里的信号量不在共享内存中,它存在于堆栈中。

所以在fork()之后,每个进程都有自己的semahore副本。您应该将semahore放在您创建的共享内存中的某个位置,以便所有进程都可以访问相同的信号量。

您可以使用sem_open()创建一个命名的semahore,而不是sem_init(),这些semahore在打开它的人之间共享。

答案 1 :(得分:0)

缺少大括号:

if((semA = sem_open("/semA123", O_CREAT, 0700, 1)) == SEM_FAILED)