使用文件夹GUID在Powershell中获取任何特殊文件夹路径

5
我希望使用PowerShell脚本来自动执行与用户登录相关的任务。
有时,用户会将其文档文件夹从默认位置移动。如何确定适用于所有特殊文件夹的PowerShell中用户文档文件夹的位置?
我尝试使用基于Lee Holmes工作的PowerShell SHGetKnownFolderPath调用。
由于pinvoke网站上的C#示例未使用StringBuilder,因此我认为它不是必需的,并剥离了builder +“-Using”引用。
param($KNOWNFOLDERID)

$Pinvoke = @’
[DllImport("shell32.dll")]
public static extern int SHGetKnownFolderPath(
     [MarshalAs(UnmanagedType.LPStruct)] Guid rfid,
     uint dwFlags,
     IntPtr hToken,
     out String pszPath  // API uses CoTaskMemAlloc
     );
‘@

$type = Add-Type -MemberDefinition $Pinvoke  
    -Name Win32Utils -Namespace SHGetKnownFolderPath  
    -PassThru

$type::SHGetKnownFolderPath($KNOWNFOLDERID) 

我也尝试使用这里的New-PInvoke,将"shell32.dll"作为lib值。

int SHGetKnownFolderPath(     [MarshalAs(UnmanagedType.LPStruct)] Guid rfid,     uint dwFlags,     IntPtr hToken,     out String pszPath  )

作为签名值,然后在PowerShell中使用。
SHGetKnownFolderPath FDD39AD0-238F-46AF-ADB4-6C85480369C7 0 0 [ref]$path

我以这个错误结束

Argument: '4' should be a System.Management.Automation.PSReference. Use [ref].
At line:1 char:56
+ New-Item Function:Global:SHGetKnownFolderPath -Value { [Microsoft.PowerShell.Com ...
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    + CategoryInfo          : NotSpecified: (:) [], MethodException
    + FullyQualifiedErrorId : NonRefArgumentToRefParameterMsg

我不明白在两次尝试中哪里出了错。 解决方案:
Kayasax 在 codeplex 上发现的 Windows7Library 二进制文件适用于像我这样的懒人,或者 David Heffernan 的脚本允许更多的自定义,但如果您使用自定义特殊文件夹,则可能需要一些手工操作,但更加可定制,两个答案都是有效的!

请注意,使用 CSIDL 的解决方案是被禁止的,因为它们无法获取 任何 特殊文件夹,CSIDL 已经过时,根据微软官方网站 http://msdn.microsoft.com/en-us/library/windows/desktop/dd378457(v=vs.85).aspx。 - Zulgrib
用户可以自行更改特殊文件夹的位置,包括文档、已保存的游戏等。如果我在一台不使用默认值的计算机上运行脚本,认为该文件夹将位于用户配置文件路径内将导致问题。我不能使用任何假定该文件夹位于默认路径上的解决方案,但这并不是问题,因为GUID解决方案将提供文件夹的完整路径。 - Zulgrib
这是针对Vista及更高版本的,对吗? - David Heffernan
这是针对可以运行Powershell V3的系统的,因为问题已经标记了它,包括Windows 7/Windows Server 2008R2及以上版本。我不知道在Vista上是否可能安装Powershell V3。 - Zulgrib
我的意思是,如果你想使用已知文件夹ID,则XP不在考虑范围内。 - David Heffernan
4个回答

5

在C#代码中包装低级别的p/invoke比直接从PowerShell尝试调用它更容易。在这里,我将pinvoke.net的声明包装成一个简单的方法,接受已知文件夹GUID并返回路径。

Add-Type @"

    using System;
    using System.Runtime.InteropServices;

    public static class KnownFolder
    {
        public static readonly Guid AddNewPrograms = new Guid( "de61d971-5ebc-4f02-a3a9-6c82895e5c04" );
        public static readonly Guid AdminTools = new Guid( "724EF170-A42D-4FEF-9F26-B60E846FBA4F" );
        public static readonly Guid AppUpdates = new Guid( "a305ce99-f527-492b-8b1a-7e76fa98d6e4" );
        public static readonly Guid CDBurning = new Guid( "9E52AB10-F80D-49DF-ACB8-4330F5687855" );
        public static readonly Guid ChangeRemovePrograms = new Guid( "df7266ac-9274-4867-8d55-3bd661de872d" );
        public static readonly Guid CommonAdminTools = new Guid( "D0384E7D-BAC3-4797-8F14-CBA229B392B5" );
        public static readonly Guid CommonOEMLinks = new Guid( "C1BAE2D0-10DF-4334-BEDD-7AA20B227A9D" );
        public static readonly Guid CommonPrograms = new Guid( "0139D44E-6AFE-49F2-8690-3DAFCAE6FFB8" );
        public static readonly Guid CommonStartMenu = new Guid( "A4115719-D62E-491D-AA7C-E74B8BE3B067" );
        public static readonly Guid CommonStartup = new Guid( "82A5EA35-D9CD-47C5-9629-E15D2F714E6E" );
        public static readonly Guid CommonTemplates = new Guid( "B94237E7-57AC-4347-9151-B08C6C32D1F7" );
        public static readonly Guid ComputerFolder = new Guid( "0AC0837C-BBF8-452A-850D-79D08E667CA7" );
        public static readonly Guid ConflictFolder = new Guid( "4bfefb45-347d-4006-a5be-ac0cb0567192" );
        public static readonly Guid ConnectionsFolder = new Guid( "6F0CD92B-2E97-45D1-88FF-B0D186B8DEDD" );
        public static readonly Guid Contacts = new Guid( "56784854-C6CB-462b-8169-88E350ACB882" );
        public static readonly Guid ControlPanelFolder = new Guid( "82A74AEB-AEB4-465C-A014-D097EE346D63" );
        public static readonly Guid Cookies = new Guid( "2B0F765D-C0E9-4171-908E-08A611B84FF6" );
        public static readonly Guid Desktop = new Guid( "B4BFCC3A-DB2C-424C-B029-7FE99A87C641" );
        public static readonly Guid Documents = new Guid( "FDD39AD0-238F-46AF-ADB4-6C85480369C7" );
        public static readonly Guid Downloads = new Guid( "374DE290-123F-4565-9164-39C4925E467B" );
        public static readonly Guid Favorites = new Guid( "1777F761-68AD-4D8A-87BD-30B759FA33DD" );
        public static readonly Guid Fonts = new Guid( "FD228CB7-AE11-4AE3-864C-16F3910AB8FE" );
        public static readonly Guid Games = new Guid( "CAC52C1A-B53D-4edc-92D7-6B2E8AC19434" );
        public static readonly Guid GameTasks = new Guid( "054FAE61-4DD8-4787-80B6-090220C4B700" );
        public static readonly Guid History = new Guid( "D9DC8A3B-B784-432E-A781-5A1130A75963" );
        public static readonly Guid InternetCache = new Guid( "352481E8-33BE-4251-BA85-6007CAEDCF9D" );
        public static readonly Guid InternetFolder = new Guid( "4D9F7874-4E0C-4904-967B-40B0D20C3E4B" );
        public static readonly Guid Links = new Guid( "bfb9d5e0-c6a9-404c-b2b2-ae6db6af4968" );
        public static readonly Guid LocalAppData = new Guid( "F1B32785-6FBA-4FCF-9D55-7B8E7F157091" );
        public static readonly Guid LocalAppDataLow = new Guid( "A520A1A4-1780-4FF6-BD18-167343C5AF16" );
        public static readonly Guid LocalizedResourcesDir = new Guid( "2A00375E-224C-49DE-B8D1-440DF7EF3DDC" );
        public static readonly Guid Music = new Guid( "4BD8D571-6D19-48D3-BE97-422220080E43" );
        public static readonly Guid NetHood = new Guid( "C5ABBF53-E17F-4121-8900-86626FC2C973" );
        public static readonly Guid NetworkFolder = new Guid( "D20BEEC4-5CA8-4905-AE3B-BF251EA09B53" );
        public static readonly Guid OriginalImages = new Guid( "2C36C0AA-5812-4b87-BFD0-4CD0DFB19B39" );
        public static readonly Guid PhotoAlbums = new Guid( "69D2CF90-FC33-4FB7-9A0C-EBB0F0FCB43C" );
        public static readonly Guid Pictures = new Guid( "33E28130-4E1E-4676-835A-98395C3BC3BB" );
        public static readonly Guid Playlists = new Guid( "DE92C1C7-837F-4F69-A3BB-86E631204A23" );
        public static readonly Guid PrintersFolder = new Guid( "76FC4E2D-D6AD-4519-A663-37BD56068185" );
        public static readonly Guid PrintHood = new Guid( "9274BD8D-CFD1-41C3-B35E-B13F55A758F4" );
        public static readonly Guid Profile = new Guid( "5E6C858F-0E22-4760-9AFE-EA3317B67173" );
        public static readonly Guid ProgramData = new Guid( "62AB5D82-FDC1-4DC3-A9DD-070D1D495D97" );
        public static readonly Guid ProgramFiles = new Guid( "905e63b6-c1bf-494e-b29c-65b732d3d21a" );
        public static readonly Guid ProgramFilesX64 = new Guid( "6D809377-6AF0-444b-8957-A3773F02200E" );
        public static readonly Guid ProgramFilesX86 = new Guid( "7C5A40EF-A0FB-4BFC-874A-C0F2E0B9FA8E" );
        public static readonly Guid ProgramFilesCommon = new Guid( "F7F1ED05-9F6D-47A2-AAAE-29D317C6F066" );
        public static readonly Guid ProgramFilesCommonX64 = new Guid( "6365D5A7-0F0D-45E5-87F6-0DA56B6A4F7D" );
        public static readonly Guid ProgramFilesCommonX86 = new Guid( "DE974D24-D9C6-4D3E-BF91-F4455120B917" );
        public static readonly Guid Programs = new Guid( "A77F5D77-2E2B-44C3-A6A2-ABA601054A51" );
        public static readonly Guid Public = new Guid( "DFDF76A2-C82A-4D63-906A-5644AC457385" );
        public static readonly Guid PublicDesktop = new Guid( "C4AA340D-F20F-4863-AFEF-F87EF2E6BA25" );
        public static readonly Guid PublicDocuments = new Guid( "ED4824AF-DCE4-45A8-81E2-FC7965083634" );
        public static readonly Guid PublicDownloads = new Guid( "3D644C9B-1FB8-4f30-9B45-F670235F79C0" );
        public static readonly Guid PublicGameTasks = new Guid( "DEBF2536-E1A8-4c59-B6A2-414586476AEA" );
        public static readonly Guid PublicMusic = new Guid( "3214FAB5-9757-4298-BB61-92A9DEAA44FF" );
        public static readonly Guid PublicPictures = new Guid( "B6EBFB86-6907-413C-9AF7-4FC2ABF07CC5" );
        public static readonly Guid PublicVideos = new Guid( "2400183A-6185-49FB-A2D8-4A392A602BA3" );
        public static readonly Guid QuickLaunch = new Guid( "52a4f021-7b75-48a9-9f6b-4b87a210bc8f" );
        public static readonly Guid Recent = new Guid( "AE50C081-EBD2-438A-8655-8A092E34987A" );
        public static readonly Guid RecycleBinFolder = new Guid( "B7534046-3ECB-4C18-BE4E-64CD4CB7D6AC" );
        public static readonly Guid ResourceDir = new Guid( "8AD10C31-2ADB-4296-A8F7-E4701232C972" );
        public static readonly Guid RoamingAppData = new Guid( "3EB685DB-65F9-4CF6-A03A-E3EF65729F3D" );
        public static readonly Guid SampleMusic = new Guid( "B250C668-F57D-4EE1-A63C-290EE7D1AA1F" );
        public static readonly Guid SamplePictures = new Guid( "C4900540-2379-4C75-844B-64E6FAF8716B" );
        public static readonly Guid SamplePlaylists = new Guid( "15CA69B3-30EE-49C1-ACE1-6B5EC372AFB5" );
        public static readonly Guid SampleVideos = new Guid( "859EAD94-2E85-48AD-A71A-0969CB56A6CD" );
        public static readonly Guid SavedGames = new Guid( "4C5C32FF-BB9D-43b0-B5B4-2D72E54EAAA4" );
        public static readonly Guid SavedSearches = new Guid( "7d1d3a04-debb-4115-95cf-2f29da2920da" );
        public static readonly Guid SEARCH_CSC = new Guid( "ee32e446-31ca-4aba-814f-a5ebd2fd6d5e" );
        public static readonly Guid SEARCH_MAPI = new Guid( "98ec0e18-2098-4d44-8644-66979315a281" );
        public static readonly Guid SearchHome = new Guid( "190337d1-b8ca-4121-a639-6d472d16972a" );
        public static readonly Guid SendTo = new Guid( "8983036C-27C0-404B-8F08-102D10DCFD74" );
        public static readonly Guid SidebarDefaultParts = new Guid( "7B396E54-9EC5-4300-BE0A-2482EBAE1A26" );
        public static readonly Guid SidebarParts = new Guid( "A75D362E-50FC-4fb7-AC2C-A8BEAA314493" );
        public static readonly Guid StartMenu = new Guid( "625B53C3-AB48-4EC1-BA1F-A1EF4146FC19" );
        public static readonly Guid Startup = new Guid( "B97D20BB-F46A-4C97-BA10-5E3608430854" );
        public static readonly Guid SyncManagerFolder = new Guid( "43668BF8-C14E-49B2-97C9-747784D784B7" );
        public static readonly Guid SyncResultsFolder = new Guid( "289a9a43-be44-4057-a41b-587a76d7e7f9" );
        public static readonly Guid SyncSetupFolder = new Guid( "0F214138-B1D3-4a90-BBA9-27CBC0C5389A" );
        public static readonly Guid System = new Guid( "1AC14E77-02E7-4E5D-B744-2EB1AE5198B7" );
        public static readonly Guid SystemX86 = new Guid( "D65231B0-B2F1-4857-A4CE-A8E7C6EA7D27" );
        public static readonly Guid Templates = new Guid( "A63293E8-664E-48DB-A079-DF759E0509F7" );
        public static readonly Guid TreeProperties = new Guid( "5b3749ad-b49f-49c1-83eb-15370fbd4882" );
        public static readonly Guid UserProfiles = new Guid( "0762D272-C50A-4BB0-A382-697DCD729B80" );
        public static readonly Guid UsersFiles = new Guid( "f3ce0f7c-4901-4acc-8648-d5d44b04ef8f" );
        public static readonly Guid Videos = new Guid( "18989B1D-99B5-455B-841C-AB7C74E4DDFC" );
        public static readonly Guid Windows = new Guid( "F38BF404-1D43-42F2-9305-67DE0B28FC23" );
    }

    public class shell32
    {
        [DllImport("shell32.dll")]
        private static extern int SHGetKnownFolderPath(
             [MarshalAs(UnmanagedType.LPStruct)] 
             Guid rfid,
             uint dwFlags,
             IntPtr hToken,
             out IntPtr pszPath
         );

         public static string GetKnownFolderPath(Guid rfid)
         {
            IntPtr pszPath;
            if (SHGetKnownFolderPath(rfid, 0, IntPtr.Zero, out pszPath) != 0)
                return ""; // add whatever error handling you fancy
            string path = Marshal.PtrToStringUni(pszPath);
            Marshal.FreeCoTaskMem(pszPath);
            return path;
         }
    }

"@

示例用法

PS C:\Users\username\Desktop> .\SHGetKnownFolderPath.ps1
PS C:\Users\username\Desktop> [shell32]::GetKnownFolderPath([KnownFolder]::Documents)
C:\Users\username\Documents
PS C:\Users\username\Desktop> [shell32]::GetKnownFolderPath([KnownFolder]::Downloads)
C:\Users\username\Downloads
PS C:\Users\username\Desktop> [shell32]::GetKnownFolderPath([KnownFolder]::InternetCache)
C:\Users\username\AppData\Local\Microsoft\Windows\Temporary Internet Files

我对PowerShell几乎一无所知,所以我相信有经验的用户能够更加自然地使用PowerShell。但是,我确定在C#代码内部处理复杂的p/invoke,并向PowerShell消费者提供更高级别的接口的基本概念是解决这个问题的正确方法。


你好,虽然在列表中使用GUID可能有效,但如果微软添加了将来的特殊文件夹,它将如何工作?这将需要脚本更新,对于系统可能具有的自定义特殊文件夹也是如此,我正确吗?无论如何,我会测试您给我的内容,因为如果我找不到需要较少维护的解决方案,那么它总比没有好。 - Zulgrib
2
您可以传递任何您喜欢的GUID。这里声明的是为了方便您使用。这展示了如何调用您问题中的函数。这是从已知文件夹GUID获取路径的方法。 - David Heffernan
艰难的选择时有发生。 - Zulgrib
请注意,注册表中“最近”文件夹的“InfoTip”值已知具有损坏的“REG_SZ”值 **@shell32,dll,-12692**。请注意,文件扩展名用逗号分隔,实际上,“InfoTip”键通常是“REG_EXPAND_SZ”。如果您想更正您的注册表,这里提供了修复方法。 - Glenn Slayden

2

您可以使用以下命令列出所有特殊文件夹:

[enum]::GetNames([System.Environment+SpecialFolder])  

如果您想获取物理路径,请使用以下方法:
[environment]::getfolderpath("mydocuments") 

请问您能否举个例子,演示如何使用“已保存的游戏”文件夹?我在列表中找不到它。 - Zulgrib
我认为保存游戏不是一个特殊的文件夹。通常它位于用户个人资料目录下($env:userprofile)。 - Loïc MICHEL
1
这是根据微软官方文档的要求,所以我询问如何通过GUID执行操作,因为我知道这个答案对于现在不再支持的XP之后由微软创建的特殊文件夹无效。 - Zulgrib
我在这里找到了一个模块,似乎可以完成你想做的事情:http://windowslibrariespsh.codeplex.com/releases/view/32315 ,你可以像这样使用它:Get-KnownFolder -KnownFolder "SavedGames" - Loïc MICHEL
我一旦有了Windows系统,就会尽快测试它,从阅读文档来看,我很可能会采用这个解决方案,请为这个解决方案提供单独的答案,以便我在测试后可以将其标记为答案。 - Zulgrib

1
我找到了一个模块,似乎可以实现你想要的功能:windowslibrariespsh.codeplex.com/releases/view/32315 你可以像这样使用它:Get-KnownFolder -KnownFolder "SavedGames"

你好,这个解决方案在x64系统上无法运行,我在CodePlex上开了一个问题,需要的库似乎是x86的。 :| - Zulgrib
成功运行了,不知道哪里出了问题。你的解决方案对我有用,但我们在一个代码网站上,我觉得接受代码作为答案比二进制更好。很难选择。 - Zulgrib

0
如果用户移动了他的文档文件夹路径,您将无法获取其位置,因为在将其移到其他位置后,它将被视为普通文件夹而不是系统文件夹。但是,如果您想要整个系统文件夹列表,可以尝试@Kayasax建议的方法。
[enum]::GetNames([System.Environment+SpecialFolder])  //for whole list

$document = [environment]::getfolderpath("mydocuments")  //for MyDocument specifically

你可以在“mydocuments”的位置上使用其他系统文件夹名称。


请问您能否举个例子,演示如何使用“已保存的游戏”文件夹?我在列表中找不到它。 - Zulgrib

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接