其他语言

本类阅读TOP10

·基于Solaris 开发环境的整体构思
·使用AutoMake轻松生成Makefile
·BCB数据库图像保存技术
·GNU中的Makefile
·射频芯片nRF401天线设计的分析
·iframe 的自适应高度
·BCB之Socket通信
·软件企业如何实施CMM
·入门系列--OpenGL最简单的入门
·WIN95中日志钩子(JournalRecord Hook)的使用

分类导航
VC语言Delphi
VB语言ASP
PerlJava
Script数据库
其他语言游戏开发
文件格式网站制作
软件工程.NET开发
信号灯之光借我一用

作者:未知 来源:月光软件站 加入时间:2005-2-28 月光软件站

??? 对于某些程序尤其是服务程序,我们一般不希望用户重复运行,禁止程序重复运行的方式有多种,在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 << "Another program with same ID is running!" << std::endl;
??? return -1;
}
if (!m_OnlyOne.Mark(20040901))
{
??? std::cout << "Create semaphore failed." << std::endl;
??? return -1;
}




相关文章

相关软件