从 powershell 递归归档某种类型的所有文件

sco*_*.se 2 windows powershell archive compress-archive

有没有办法使用压缩存档脚本,当从路径运行时:

  1. 归档与通配符过滤器匹配的所有文件(例如 *.doc)
  2. 将此类文件存档在当前文件夹和所有子文件夹中
  3. 保存相对文件夹结构(不过,使用相对或绝对的选项会很好)

我很难让它同时完成这三件事。

编辑:

以下过滤和递归,但不维护文件夹结构

Get-ChildItem -Path ".\" -Filter "*.docx" -Recurse |
Compress-Archive -CompressionLevel Optimal -DestinationPath "$pwd\doc.archive-$(Get-Date -f yyyyMMdd.hhmmss).zip"
Run Code Online (Sandbox Code Playgroud)

此项不递归:

Compress-Archive -Path "$pwd\*.docx" -CompressionLevel Optimal -DestinationPath "$pwd\doc.archive-$(Get-Date -f yyyyMMdd.hhmmss).zip"
Run Code Online (Sandbox Code Playgroud)

在某些时候,我有一个命令会递归但不会过滤,但现在无法返回。

mkl*_*nt0 5

不幸的是,从 Windows PowerShell v5.1 / PowerShell Core 6.1.0 开始,Compress-Archive它的功能非常有限:

  • 保留子目录树的唯一方法是将目录路径传递给Compress-Archive.

    • 不幸的是,这样做没有提供包含/排除机制来仅选择文件的子集。

    • 此外,生成的存档将在内部包含以输入目录命名的单个根目录(例如,如果传递C:\temp\fooCompress-Archive,生成的存档将包含一个包含foo输入目录子树的目录 - 与包含在顶层C:\temp\foo内容相反)。

    • 没有保留绝对路径的选项。

  • 一个麻烦的解决方法是创建目录树的临时副本,仅包含感兴趣的文件Copy-Item -Recurse -Filter *.docx . $env:TEMP\tmpDir; Compress-Archive $env:TEMP\tmpDir out.zip- 请注意,将包括空目录)

    • 鉴于您仍然总是会得到一个以存档内的输入目录命名的根目录,即使这可能对您不起作用 - 请参阅底部的替代方案。

您可能有更好的选择


直接使用.NET v4.5+类解决问题[System.IO.Compression.ZipFile]

笔记:

  • 在 Windows PowerShell 中,与 PowerShell Core不同,您大多数情况下使用 手动加载相关程序集Add-Type -AssemblyName System.IO.Compression.FileSystem

  • 由于从 Windows PowerShell v5.1/PowerShell Core 6.1.0 开始,PowerShell 不支持隐式[System.IO.Compression.ZipFileExtensions]使用扩展方法,因此您还必须显式使用该类。

# Windows PowerShell: must load assembly System.IO.Compression.FileSystem manually.
Add-Type -AssemblyName System.IO.Compression.FileSystem

# Create the target archive via .NET to provide more control over how files
# are added.
# Make sure that the target file doesn't already exist.
$archive = [System.IO.Compression.ZipFile]::Open(
  "$pwd\doc.archive-$(Get-Date -f yyyyMMdd.hhmmss).zip",
  'Create'
)

# Get the list of files to archive with their relative paths and
# add them to the target archive one by one.
$useAbsolutePaths = $False # Set this to true to use absolute paths instead.
Get-ChildItem -Recurse -Filter *.docx | ForEach-Object {
    # Determine the entry path, i.e., the archive-internal path.
    $entryPath = (
          ($_.FullName -replace ([regex]::Escape($PWD.ProviderPath) + '[/\\]'), ''), 
          $_.FullName
        )[$useAbsolutePaths]
    $null = [System.IO.Compression.ZipFileExtensions]::CreateEntryFromFile(
      $archive, 
      $_.FullName, 
      $entryPath
    )
  }

# Close the archive.
$archive.Dispose()
Run Code Online (Sandbox Code Playgroud)