5

我必须开发一个解析日志文件并将特定数据发送到服务器的应用程序。它必须在 Linux 和 Windows 上运行。

当我想测试日志滚动系统时出现问题(它将 .1 附加到创建一个具有相同名称的新系统的名称)。在 Windows 上(尚未在 Linux 上测试)我无法重命名使用 std::ifstream() 打开的文件(独占访问?)即使我以“输入模式”(ios::in)打开它.

是否有跨平台的方式以非独占方式打开文件?

4

5 回答 5

3

有没有办法以非独占方式打开文件,

是的,使用 Win32,将各种 FILE_SHARE_Xxxx 标志传递给 CreateFile。

它是跨平台的吗?

不,它需要特定于平台的代码。

由于令人讨厌的向后兼容性问题(DOS 应用程序,作为单任务,假设没有任何东西可以从它们下面删除文件,即它们可以 fclose() 然后 fopen() 没有任何问题;Win16 保留了这个假设以使移植 DOS 应用程序更容易,Win32 保留了这个假设,使移植 Win16 应用程序更容易,这很糟糕),Windows 默认以独占方式打开文件。

底层操作系统基础架构支持删除/重命名打开的文件(尽管我认为它确实有不能删除内存映射文件的限制,我认为这不是在 *nix 上发现的限制),但默认的打开语义没有。

C++ 对此一无所知。C++操作环境与DOS操作环境大体相同——没有其他应用程序同时运行,因此无需控制文件共享。

于 2008-08-26T11:02:41.227 回答
1

需要独占模式的不是读取操作,而是重命名,因为这本质上与将文件移动到新位置相同。

我不确定,但我不认为这可以做到。尝试复制文件,然后在不再读取时删除/替换旧文件。

于 2008-08-26T10:28:34.663 回答
1

Win32 文件系统语义要求重命名的文件在重命名时不打开(以任何模式)。您需要关闭该文件,重命名它,然后创建新的日志文件。

Unix 文件系统语义允许您重命名打开的文件,因为文件名只是指向 inode 的指针。

于 2008-08-26T10:32:45.947 回答
1

如果您只是从文件中读取,我知道可以使用 windows api CreateFile 来完成。只需指定 FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE 作为 dwShareMode 的输入。

不幸的是,这不是跨平台的。但 Linux 可能也有类似的东西。

有关 CreateFile 的更多信息,请参阅 msdn

编辑:只是关于 Greg Hewgill 评论的简短说明。我刚刚测试了 FILE_SHARE* 的东西(100% 肯定)。如果您以只读方式打开并指定 FILE_SHARE* 参数,则可以在 Windows 中删除和重命名文件。

于 2008-08-26T10:35:41.120 回答
0

我会确保你不要让文件保持打开状态。例如,如果您的应用程序崩溃,这会导致奇怪的事情。我会做什么:

  1. 将(读取/写入/滚动到新文件)抽象到一个类中,并在您想要滚动到该类中的新文件时安排关闭文件。(这是最简洁的方法,因为您已经有了翻转代码,所以您已经完成了一半。)
  2. 如果您必须有多个读/写访问点,需要 fstreams 的所有功能并且不想编写完整的包装器,那么我能想到的唯一跨平台解决方案是在不需要时始终关闭文件,并让翻转代码在放弃前需要翻转时尝试几次获得对文件的独占访问权限。
于 2008-09-16T11:47:16.700 回答