31

我希望我的应用程序存储一些数据以供所有用户访问。使用 Python,我怎样才能找到数据应该去哪里?

4

6 回答 6

42

如果您不想为诸如 winpaths 之类的第三方模块添加依赖项,我建议您使用 Windows 中已有的环境变量:

具体来说,您可能想要ALLUSERSPROFILE获取通用用户配置文件文件夹的位置,即应用程序数据目录所在的位置。

例如:

C:\> python -c "import os; print os.environ['ALLUSERSPROFILE']"
C:\Documents and Settings\All Users

编辑:查看 winpaths 模块,它使用 ctypes,所以如果您只想使用代码的相关部分而不安装 winpath,您可以使用它(显然为简洁起见省略了一些错误检查)。

import ctypes
from ctypes import wintypes, windll

CSIDL_COMMON_APPDATA = 35

_SHGetFolderPath = windll.shell32.SHGetFolderPathW
_SHGetFolderPath.argtypes = [wintypes.HWND,
                            ctypes.c_int,
                            wintypes.HANDLE,
                            wintypes.DWORD, wintypes.LPCWSTR]


path_buf = wintypes.create_unicode_buffer(wintypes.MAX_PATH)
result = _SHGetFolderPath(0, CSIDL_COMMON_APPDATA, 0, 0, path_buf)
print path_buf.value

示例运行:

C:\> python get_common_appdata.py
C:\Documents and Settings\All Users\Application Data
于 2009-03-09T16:12:41.823 回答
15

来自http://snipplr.com/view.php?codeview&id=7354

homedir = os.path.expanduser('~')

# ...works on at least windows and linux. 
# In windows it points to the user's folder 
#  (the one directly under Documents and Settings, not My Documents)


# In windows, you can choose to care about local versus roaming profiles.
# You can fetch the current user's through PyWin32.
#
# For example, to ask for the roaming 'Application Data' directory:
#  (CSIDL_APPDATA asks for the roaming, CSIDL_LOCAL_APPDATA for the local one)
#  (See microsoft references for further CSIDL constants)
try:
    from win32com.shell import shellcon, shell            
    homedir = shell.SHGetFolderPath(0, shellcon.CSIDL_APPDATA, 0, 0)

except ImportError: # quick semi-nasty fallback for non-windows/win32com case
    homedir = os.path.expanduser("~")

要获取所有用户的 app-data 目录,而不是当前用户,只需使用shellcon.CSIDL_COMMON_APPDATA而不是shellcon.CSIDL_APPDATA.

于 2009-09-19T09:57:14.897 回答
10

看看http://ginstrom.com/code/winpaths.html。这是一个简单的模块,它将检索 Windows 文件夹信息。该模块实现get_common_appdata为所有用户获取 App Data 文件夹。

于 2009-03-09T15:58:23.700 回答
3

由于与非美国版本的 Windows 和 Vista 不兼容,先前的答案被删除。

编辑: 要扩展 Out Into Space 的答案,您将使用该 winpaths.get_common_appdata功能。easy_install winpaths您可以使用或转到 pypi 页面http://pypi.python.org/pypi/winpaths/并下载 .exe 安装程序来获取 winpath 。

于 2009-03-09T15:59:26.303 回答
3

您可以使用模块os.environ中的字典访问所有操作系统环境变量os。但是,从该字典中选择使用哪个键可能很棘手。特别是,在使用这些路径时,您应该注意国际化(即非英语)版本的 Windows。

os.environ['ALLUSERSPROFILE']应该为您提供计算机上所有用户的根目录,但之后请注意不要硬编码子目录名称,如“应用程序数据”,因为这些目录在非英语版本的 Windows 上不存在。就此而言,您可能希望对可以期望设置 ALLUSERSPROFILE 环境变量的 Windows 版本进行一些研究(我自己也不知道——它可能是通用的)。

我这里的 XP 机器有一个 COMMONAPPDATA 环境变量,它指向 All Users\Application Data 文件夹,但是我的 Win2K3 系统没有这个环境变量。

于 2009-03-09T16:51:46.427 回答
1

由于不推荐使用SHGetFolderPath,因此您还可以在 Vista 和更新版本中使用SHGetKnownFolderPath 。这也使您可以查找比 SHGetFolderPath 更多的路径。这是一个精简的示例(完整代码可在 Gist 上找到):

import ctypes, sys
from ctypes import windll, wintypes
from uuid import UUID

class GUID(ctypes.Structure):   # [1]
    _fields_ = [
        ("Data1", wintypes.DWORD),
        ("Data2", wintypes.WORD),
        ("Data3", wintypes.WORD),
        ("Data4", wintypes.BYTE * 8)
    ] 

    def __init__(self, uuid_):
        ctypes.Structure.__init__(self)
        self.Data1, self.Data2, self.Data3, self.Data4[0], self.Data4[1], rest = uuid_.fields
        for i in range(2, 8):
            self.Data4[i] = rest>>(8 - i - 1)*8 & 0xff

class FOLDERID:     # [2]
    LocalAppData            = UUID('{F1B32785-6FBA-4FCF-9D55-7B8E7F157091}')
    LocalAppDataLow         = UUID('{A520A1A4-1780-4FF6-BD18-167343C5AF16}')
    RoamingAppData          = UUID('{3EB685DB-65F9-4CF6-A03A-E3EF65729F3D}')

class UserHandle:   # [3]
    current = wintypes.HANDLE(0)
    common  = wintypes.HANDLE(-1)

_CoTaskMemFree = windll.ole32.CoTaskMemFree     # [4]
_CoTaskMemFree.restype= None
_CoTaskMemFree.argtypes = [ctypes.c_void_p]

_SHGetKnownFolderPath = windll.shell32.SHGetKnownFolderPath     # [5] [3]
_SHGetKnownFolderPath.argtypes = [
    ctypes.POINTER(GUID), wintypes.DWORD, wintypes.HANDLE, ctypes.POINTER(ctypes.c_wchar_p)
] 

class PathNotFoundException(Exception): pass

def get_path(folderid, user_handle=UserHandle.common):
    fid = GUID(folderid) 
    pPath = ctypes.c_wchar_p()
    S_OK = 0
    if _SHGetKnownFolderPath(ctypes.byref(fid), 0, user_handle, ctypes.byref(pPath)) != S_OK:
        raise PathNotFoundException()
    path = pPath.value
    _CoTaskMemFree(pPath)
    return path

common_data_folder = get_path(FOLDERID.RoamingAppData)

# [1] http://msdn.microsoft.com/en-us/library/windows/desktop/aa373931.aspx
# [2] http://msdn.microsoft.com/en-us/library/windows/desktop/dd378457.aspx
# [3] http://msdn.microsoft.com/en-us/library/windows/desktop/bb762188.aspx
# [4] http://msdn.microsoft.com/en-us/library/windows/desktop/ms680722.aspx
# [5] http://www.themacaque.com/?p=954
于 2013-11-19T19:05:26.463 回答