web-dev-qa-db-ja.com

Powershellまたはコマンドラインを使用して、Windowsで圧縮/圧縮フォルダーを作成する

毎晩データベーススキーマファイルを作成していますが、毎晩作成されたすべてのファイル(データベースごとに1つ)をフォルダーに入れて、そのフォルダーを圧縮したいと思います。スキーマを作成するPowerShellスクリプトがあります。データベースの作成スクリプトのみを作成し、すべてのファイルを新しいフォルダーに追加します。問題は、このプロセスの圧縮部分にあります。

フォルダーの圧縮を処理するプレインストールされたWindowsユーティリティでこれを達成できるかどうか、誰にも考えがありますか?

7Zipのようなものではなく、可能であればそのユーティリティを使用するのが最善です(すべての顧客のサーバーに7Zipをインストールする気はありません。

23
TechDawg270

次に、拡張機能に依存しないZip関連の関数をいくつか示します。 Windows PowerShellでファイルを圧縮する

あなたが興味を持ちそうな主な機能は次のとおりです。

_function Add-Zip
{
    param([string]$zipfilename)

    if(-not (test-path($zipfilename)))
    {
        set-content $zipfilename ("PK" + [char]5 + [char]6 + ("$([char]0)" * 18))
        (dir $zipfilename).IsReadOnly = $false  
    }

    $shellApplication = new-object -com Shell.application
    $zipPackage = $shellApplication.NameSpace($zipfilename)

    foreach($file in $input) 
    { 
            $zipPackage.CopyHere($file.FullName)
            Start-sleep -milliseconds 500
    }
}
_

使用法:

_dir c:\demo\files\*.* -Recurse | Add-Zip c:\demo\myzip.Zip
_

注意点が1つあります。_Shell.application_オブジェクトのNameSpace()関数は、パスが絶対パスでない場合、書き込み用のZipファイルを開くことができません。したがって、_Add-Zip_に相対パスを渡すと、nullエラーで失敗するため、Zipファイルへのパスは絶対パスでなければなりません。

または、関数の先頭に_$zipfilename = resolve-path $zipfilename_を追加することもできます。

20
voithos

最新の.NET 4.5フレームワークのネイティブな方法ですが、完全に機能がありません。

作成:

Add-Type -Assembly "System.IO.Compression.FileSystem" ;
[System.IO.Compression.ZipFile]::CreateFromDirectory("c:\your\directory\to\compress", "yourfile.Zip") ;

抽出:

Add-Type -Assembly "System.IO.Compression.FileSystem" ;
[System.IO.Compression.ZipFile]::ExtractToDirectory("yourfile.Zip", "c:\your\destination") ;

前述のように、完全に機能がないため、上書きフラグを期待しないでください。

23
sonjz

PowersShell 5では、すぐにタスクを実行する Compress-Archive コマンドレットがあります。

7

これは、例に従って、System.IO.Packaging.ZipPackageで。\ inの内容を。\ out.Zipに圧縮します here

$zipArchive = $pwd.path + "\out.Zip"
[System.Reflection.Assembly]::Load("WindowsBase,Version=3.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35")
$ZipPackage=[System.IO.Packaging.ZipPackage]::Open($zipArchive, [System.IO.FileMode]"OpenOrCreate", [System.IO.FileAccess]"ReadWrite")
$in = gci .\in | select -expand fullName
[array]$files = $in -replace "C:","" -replace "\\","/"
ForEach ($file In $files) {
   $partName=New-Object System.Uri($file, [System.UriKind]"Relative")
   $part=$ZipPackage.CreatePart($partName, "application/Zip", [System.IO.Packaging.CompressionOption]"Maximum")
   $bytes=[System.IO.File]::ReadAllBytes($file)
   $stream=$part.GetStream()
   $stream.Write($bytes, 0, $bytes.Length)
   $stream.Close()
                                                    }
$ZipPackage.Close()
2
noam

PowershellでZipファイルに対するvoithosの回答を使用しましたが、Add-Zip機能に1つの問題がありました。Start-sleep-milliseconds 500は、その時間にファイルを完全に圧縮できなかった場合に問題を引き起こしました->次のもの完了する前にエラーが発生し、一部のファイルが圧縮されませんでした。

そのため、少し遊んだ後、最初に$ zipPackage.Items()のカウントをチェックするカウンターを取得し、アイテムのカウントが増加した後にのみ続行しようとします(場合によっては0を返すため、動作しませんでした)すべきではありません)パッケージがまだファイルを圧縮/コピーしている場合は0を返すことがわかりました(ハハ)内部にstart-sleepを含む単純なwhileループを追加しました。続行する前にzipPackage.Items()。countがゼロ以外の値になるのを待って、これで問題が解決したようです。

function Add-Zip
{
param([string]$zipfilename)

if(-not (test-path($zipfilename)))
{
    set-content $zipfilename ("PK" + [char]5 + [char]6 + ("$([char]0)" * 18))
    (dir $zipfilename).IsReadOnly = $false  
}

$shellApplication = new-object -com Shell.application
$zipPackage = $shellApplication.NameSpace($zipfilename)

foreach($file in $input) 
{ 
        $zipPackage.CopyHere($file.FullName)
        do
        {
            Start-sleep -milliseconds 250
        }
        while ($zipPackage.Items().count -eq 0)
}
}
2
Verts