信号灯之光借我一用

王朝other·作者佚名  2006-01-09
宽屏版  字体: |||超大  

??? 对于某些程序尤其是服务程序,我们一般不希望用户重复运行,禁止程序重复运行的方式有多种,在linux下我选择利用信号量来禁止程序多次运行。

??? 原理:程序执行时在内存中创建一个特定Key的信号量,并对该信号量的值做加1计数操作,并设置为进程退出自动减1。因此程序在启动时判断这个特定Key的信号量的值是否大于0,如大于0则表示已有实例在运行,如等于0或不存在这个信号量则说明尚没有实例运行。

const int SEM_PRMS = 0644;//信号量操作权限,0644即主用户(属主)可读写、组成员及其它成员可读不可写

class COnlyOne

{

public:

??? bool Exists(key_t _nKey);?//检查信号量是否已存在

??? bool Mark(key_t _nKey);??//设置信号量

??? bool Unmark(key_t _nKey);?//清除信号量

};

bool gnl_lnx::COnlyOne::Exists(key_t _nKey)

{

??? int nID = semget(_nKey, 0, 0);

??? if (nID == -1) return false;

??? return semctl(nID, 0, GETVAL, NULL) 0;

}

bool gnl_lnx::COnlyOne::Mark(key_t _nKey)

{

??? int nID = semget(_nKey, 0, 0);

??? if (nID == -1)

??????? nID = semget(_nKey, 1, IPC_CREAT | IPC_EXCL | SEM_PRMS);

?

??? struct sembuf buf[2];

??? buf[0].sem_num = 0;

??? buf[0].sem_op = 0;

??? buf[0].sem_flg = IPC_NOWAIT;

??? buf[1].sem_num = 0;

??? buf[1].sem_op = 1;

??? buf[1].sem_flg = SEM_UNDO;? //进程退出时自动回滚

??? return semop(nID, &buf[0], 2) == 0;

}

bool gnl_lnx::COnlyOne::Unmark(key_t _nKey)

{

??? int nID = semget(_nKey, 0, 0);

??? if (nID == -1) return true;

??? return semctl(nID, 0, IPC_RMID, 0) == 0;

}

调用示例:

if (m_OnlyOne.Exists(20040901))

{

??? std::cout

??? return -1;

}

if (!m_OnlyOne.Mark(20040901))

{

??? std::cout

??? return -1;

}

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
© 2005- 王朝网络 版权所有