2

这就是问题所在:我不喜欢我的程序的多个实例,这就是我禁用它们的原因。我的程序会打开一个特定的 mime 类型。在我的系统(Ubuntu 12.04)中,当我双击这些文件之一时,将执行以下操作:

/usr/bin/myprogram /path/to/double/clicked/file.myextension

正如我所说,我不喜欢多个实例,因此,如果程序已经在运行并且用户选择打开其中一个文件,则会向已经存在的实例发送一条 DBus 消息,以便处理打开的文件。因此,如果有一个已经在运行的实例并且用户选择了 3 个文件来用我的程序打开并点击 [Enter] 按钮,系统将执行:

/usr/bin/myprogram /path/to/double/clicked/file1.myextension
/usr/bin/myprogram /path/to/double/clicked/file2.myextension
/usr/bin/myprogram /path/to/double/clicked/file3.myextension

所有这些实例都检测到已经运行的实例并将打开的文件发送给它。完全没有问题,直到现在。

但是,如果没有一个已经在运行的实例并且用户选择用我的程序一起打开 3 个文件怎么办?系统将再次同时调用:

/usr/bin/myprogram /path/to/double/clicked/file1.myextension
/usr/bin/myprogram /path/to/double/clicked/file2.myextension
/usr/bin/myprogram /path/to/double/clicked/file3.myextension

并且这些实例中的每一个都会意识到有一个已经在运行的实例,它会尝试向已经运行的实例发送一条 DBus 消息并退出。因此,所有 3 个进程都将执行相同的操作,并且不会运行任何内容。

我怎样才能避免这个问题?

PS:为了检测是否已经有正在运行的实例,我实现了以下代码:

bool already_runs(){
return !system("pidof myprogram | grep \" \" > /dev/null");
}
4

2 回答 2

0

我会使用一些共享内存来存储第一个进程的 pid。QSharedMemory类将在这里为您提供帮助

您的程序应该做的第一件事是尝试创建一个共享内存段(使用您自己制作的密钥)并将您的 pid 存储在其中。如果创建调用失败,那么您可以尝试附加到该段。如果成功,那么您可以从中读取原始进程的 pid。

编辑:另外,请记住在写入或读取共享内存之前使用 lock(),然后在完成后调用 unlock()。

于 2012-07-24T14:40:13.247 回答
0

在 DBus 中执行此操作的标准方法是在总线上获取应用程序的名称;一个实例将赢得比赛并成为运行实例。

但是,您应该能够使用 Qt 功能来做到这一点,这将更好地与您的应用程序的其余部分集成;请参阅Qt:单实例应用程序保护的最佳实践

于 2012-07-24T15:05:40.113 回答