1

我有一个项目,我必须使用 SYS V 信号量。我有几个进程共享一个信号量(使用相同的键)并使用以下代码对其进行初始化:

bool semaphore_init(semaphore_id_t* sem, int sem_value, key_t key)
{
    /* Try to get a semaphore, to check if you will be an owner */
    *sem = semget(key, 1, IPC_CREAT | IPC_EXCL | 0666);
    if (*sem == -1)
    {
        if (errno == EEXIST)
        {
            /* We are not owners, get semaphore without exclusive flag */
            *sem = semget(key, 1, IPC_CREAT | 0666);
            if (*sem == -1) return false;
        }
        else return false;
    }
    else
    {
        /* We are owners, initialize semaphore */
        int return_value = semctl(*sem , 0, SETVAL, sem_value);
        if (return_value == -1) return false;
    }

    return true;
}

我的问题是:我想在所有使用它的进程终止时删除这个信号量。使用:

semctl(*sem, 0, IPC_RMID)

不是一种选择。它立即删除信号量,其他进程得到未定义的行为。我只是找不到使用 SYS V API 的正确方法。

4

1 回答 1

0

您可以使用引用计数。只需让每个进程在信号量初始化后递增 ref 计数器,并在进程终止时递减它。最后一个将其减为 0 的也将删除它,或者当它知道所有其他进程已终止时让主进程清理它。

您可能还需要另一种锁定机制来同步对引用计数器的访问,您可能会在生成并等待其他进程的主进程中创建/删除它。

此外,如果您的进程可能异常终止,请注意悬空引用。

于 2013-01-16T14:32:24.617 回答