【发布时间】:2013-10-17 07:57:51
【问题描述】:
我想读/写一个文件,但允许它被其他进程删除/重命名。在C# 中,您可以使用FileShare.Delete 打开文件,Python 是否有等效项?
【问题讨论】:
我想读/写一个文件,但允许它被其他进程删除/重命名。在C# 中,您可以使用FileShare.Delete 打开文件,Python 是否有等效项?
【问题讨论】:
如果您想要一个跨平台的等价物……真的没有。在 POSIX 系统上,其他进程始终可以删除/重命名您打开的文件*,除非您竭尽全力阻止它。因此,您只需要在 Windows 上执行此操作,几乎无处不在
Python 的标准文件对象不允许您直接控制 Windows 共享标志。 (这是因为它们使用跨平台 API,例如 stdio 的 fopen,而不是 Windows 特定的 API。)
如果你想这样做,你必须调用不同的文件函数。
在 IronPython 中,您当然可以调用与 C# 中完全相同的 .NET 函数。
在 CPython 中,您不能使用 .NET,因为 CPython 是本机 Win32 应用程序。这意味着您必须致电CreateFile。您可以使用pywin32 包来简化此操作,但这仍然有点麻烦。
您需要这样的东西(未经测试),而不仅仅是f = open(path, 'r+'):
f = win32file.CreateFile(path,
win32con.GENERIC_READ | win32con.GENERIC_WRITE,
win32con.FILE_SHARE_DELETE,
None,
0,
win32con.OPEN_ALWAYS,
None)
然后,而不是 buf = f.read(4096),您可以:
buflen, buf = win32file.ReadFile(f, 4096, None)
而且,您必须明确地关闭它,而不是仅仅将其粘贴在 with 语句中:
win32file.CloseFile(f)
如果您打算做很多这样的事情,将win32file 句柄封装在类似文件的对象中并不难,而且可能值得这样做。
最棘手的一点是决定您希望与open 的参数保持多近,并编写代码将您获得的值转换为CreateFile 想要的值。 (也有一些边缘情况,Win32 本机文件处理方式与 Win32 stdio 文件不同,但通常这些不会让你感到厌烦。)
您可能还想查看 io 模块中的帮助程序,这些帮助程序让您只需实现少数原始的打开/读取/写入/关闭函数并自动将其包装在一个完整的类似文件的对象中。
* 实际上,他们所能做的就是取消链接您打开的目录条目。即使这是链接到该文件的唯一目录条目,该文件本身仍然存在,直到它的最后一个打开句柄被关闭。
【讨论】:
open,而不是 stdio API。从理论上讲,这应该更容易改变为使用 Win32 API,但这将是一个相当大的改变;除非您可以在任何地方用 HANDLE 替换 fd,并为 MSVCRT 在 fd 之上模拟的所有内容编写新代码,否则它不能成为直接替换,而只是一个单独的类似文件的类型。但我有点惊讶没有人使用 PyPI 开发那种单独的类型……