1

所以,我有一个 Linux 进程,我试图在其中管理磁带上的一些文件。我有以下代码尝试从磁带上的当前存档中提取文件 catalog.xml 并将其复制到固定位置(最终,我将解析该文件并对结果进行一些处理)。但是,我的代码间歇性地无法正常工作。tar 命令总是成功,(如果我检查文件系统,我会看到 catalog.xml 文件),但有时我的后续检查以查看文件是否存在会返回 false。我在做一些明显错误的事情吗?似乎我可能在 fork()ed 进程返回和该进程的结果在文件系统上可见之间遇到了竞争条件 - 我需要进行一些调用吗?

  pid_t tPid = vfork();
  if (0 == tPid)
  {
    int tChildRet;
    tChildRet = execlp("tar", "tar", "-xvf", "/dev/nst0", "-C", "/tmp", "catalog.xml", (char *) 0);
    _exit(-1 == tChildRet ? EXIT_FAILURE : EXIT_SUCCESS);
  }
  else
  {
    wait(&ret);
  }
std::ifstream tCatalogFile("/tmp/catalog.xml");
if (tCatalogFile)
{
   cout << "File exists!" << endl;
} else {
   cout << "File does not exist!" << endl;
}

而且我得到“文件存在!” 或“文件不存在!”,似乎是随机的。

其他说明: 关于失败案例:

  • 如果我执行 a stat ("/tmp/catalog.xml"),我会得到 -1 的返回值,并将 errno 设置为 ENOENT。
  • tar 命令(使用 -v 标志运行)产生预期的一行输出(“catalog.xml”)

/tmp 是本地 tmpfs 文件系统;磁带机是本地设备。我在 x86_64 机器上使用带有 g++ 4.1.2 的 2.6.30.9 Linux 内核。

提前致谢!

4

2 回答 2

3

尝试在父调用syncwait调用。

如果这不起作用,您可能需要循环和/或睡眠,直到父母可以打开文件,因为您知道它在那里。

于 2011-02-14T20:45:22.307 回答
2

如果execlp成功,它将永远不会到达您调用的行_exit。您没有检查ret来自wait. 目前尚不清楚您为什么应该使用vfork. 等等。

既然父母除了等待孩子完成之外没有做任何其他事情,为什么不让你的生活更轻松,只使用system()?

于 2011-02-14T20:47:47.647 回答