-1

在 python 中,我如何识别一个文件是“窗口系统文件”。在命令行中,我可以使用以下命令执行此操作:

ATTRIB "c:\file_path_name.txt"

如果返回有“S”字符,那么它是一个windows系统文件。我无法弄清楚python中的等价物。一些类似查询的示例如下所示:

文件是可写的吗?

import os

filePath = r'c:\testfile.txt'

if os.access(filePath, os.W_OK):
   print 'writable'
else:
   print 'not writable'

其他方式...

import os
import stat

filePath = r'c:\testfile.txt'

attr = os.stat(filePath)[0]
if not attr & stat.S_IWRITE:
   print 'not writable'
else:
   print 'writable'

但我找不到识别 Windows 系统文件的函数或枚举。希望有一种内置的方法可以做到这一点。我宁愿不必使用 win32com 或其他外部模块。

我想这样做的原因是因为我正在使用 os.walk 将文件从一个驱动器复制到另一个驱动器。如果有一种方法可以遍历目录树,同时忽略可能也可以工作的系统文件。

谢谢阅读。



这是我根据答案提出的解决方案:

使用win32api:

import win32api
import win32con

filePath = r'c:\test_file_path.txt'

if not win32api.GetFileAttributes(filePath) & win32con.FILE_ATTRIBUTE_SYSTEM:
   print filePath, 'is not a windows system file'
else:
   print filePath, 'is a windows system file'

并使用 ctypes:

import ctypes
import ctypes.wintypes as types

# From pywin32
FILE_ATTRIBUTE_SYSTEM = 0x4

kernel32dll = ctypes.windll.kernel32


class WIN32_FILE_ATTRIBUTE_DATA(ctypes.Structure):
   _fields_ = [("dwFileAttributes", types.DWORD),
               ("ftCreationTime", types.FILETIME),
               ("ftLastAccessTime", types.FILETIME),
               ("ftLastWriteTime", types.FILETIME),
               ("nFileSizeHigh", types.DWORD),
               ("nFileSizeLow", types.DWORD)]

def isWindowsSystemFile(pFilepath):
   GetFileExInfoStandard = 0

   GetFileAttributesEx = kernel32dll.GetFileAttributesExA
   GetFileAttributesEx.restype = ctypes.c_int
   # I can't figure out the correct args here
   #GetFileAttributesEx.argtypes = [ctypes.c_char, ctypes.c_int, WIN32_FILE_ATTRIBUTE_DATA]

   wfad = WIN32_FILE_ATTRIBUTE_DATA()
   GetFileAttributesEx(pFilepath, GetFileExInfoStandard, ctypes.byref(wfad))

   return wfad.dwFileAttributes & FILE_ATTRIBUTE_SYSTEM

filePath = r'c:\test_file_path.txt'

if not isWindowsSystemFile(filePath):
   print filePath, 'is not a windows system file'
else:
   print filePath, 'is a windows system file'

我想知道在我的代码中粘贴常量“FILE_ATTRIBUTE_SYSTEM”是否合法,或者我也可以使用 ctypes 获得它的值?

4

1 回答 1

1

但我找不到识别 Windows 系统文件的函数或枚举。希望有一种内置的方法可以做到这一点。

哪有这回事。Python 的文件抽象没有任何“系统文件”的概念,所以它没有给你任何获取它的方法。此外,Python是微软 C 运行时库中的or函数的stat一个非常薄的包装器,它没有任何“系统文件”的概念。这样做的原因是 Python 文件和微软的 C 库都被设计为“非常像 POSIX”。stat_stat

当然,Windows对文件有完全不同的抽象。但是这个没有被 , 等函数暴露open出来stat;相反,有一组完全并行的函数,如CreateFile,GetFileAttributes等。如果你想要这些信息,你必须调用它们。

我宁愿不必使用 win32com 或其他外部模块。

好吧,您不需要win32com,因为这只是 Windows API,而不是 COM。

但这win32api是最简单的方法。它提供了一个很好的包装器GetFileAttributesEx,这是您要调用的函数。

如果您不想使用外部模块,则始终可以通过调用 Windows API 函数ctypes。或者subprocess用来运行命令行工具(比如ATTRIB——或者,如果你愿意,比如DIR /S /A-S让 Windows 为你做 recursive-walk-skipping-system-files 位……)。

文档展示了如何调用 Windows API 函数,但第ctypes一次有点棘手。

首先你需要去MSDN页面找出你需要加载什么DLL(kernel32),你的函数是否有单独的A和W变体(它有),以及为任何常量传递什么值(你必须遵循一个指向另一个页面的链接,并且知道 C 枚举是如何工作的,找出它GetFileExInfoStandard是 0),然后你需要弄清楚如何定义任何struct必要的 s。在这种情况下,是这样的:

from ctypes import *
kernel = windll.kernel32

GetFileExInfoStandard = 0

GetFileAttributesEx = kernel.GetFileAttributesEx
GetFileAttributesEx.restype = c_int
GetFileAttributesEx.argypes = # ...

如果您真的想避免使用win32api,您可以自己完成ctypes包装器的工作。就个人而言,我会使用win32api.


同时:

我想这样做的原因是因为我正在使用 os.walk 将文件从一个驱动器复制到另一个驱动器。如果有一种方法可以遍历目录树,同时忽略可能也可以工作的系统文件。

对于这种情况,尤其是考虑到您抱怨检查每个文件太慢时,您可能都不想使用os.walk。相反,使用FindFirstFileEx, 并手动进行递归。您可以区分文件和目录,而不必stat(或GetFileAttributesEx)每个文件(os.walk在幕后进行),您可以直接在 find 函数中过滤掉系统文件,而不必过滤stat每个文件等。

同样,选项是相同的:win32api如果您希望它简单,则使用,ctypes否则使用。

但在这种情况下,我会看一下 Ben Hoyt 的betterwalk,因为他已经完成ctypes了您想要的 99% 的 -wrapping 和 95% 的其余代码。

于 2013-05-01T21:07:16.653 回答