Windows小技巧 -- 批處理文件實現目錄下文件批量打包壓縮

最近整理資料,發現很多 pdf 文檔,佔用了不少的存儲空間,考慮使用 7-zip 進行壓縮存儲,由於文件比較多,且分散在不同目錄下,一個個文件壓縮比較繁瑣。爲了提高效率且,方便後面取用,所有打算搗鼓個批處理文件,來實現命令行調用 7-zip 來對目錄下的每個文檔進行單獨打包,並將打包後的壓縮包存到指定的目錄下,詳細內容見下文。

本文在 Windows10 64位 操作系統下,使用的命令行提示符(cmd)版本爲 10.0.17134.7067-zip64位 16.04 版本,使用下面腳本時請確保安裝了 7-zip ,可在此處下載最新版


for 命令實現

以下展示幾種通過 for 命令來遍歷打包壓縮批處理文件的示例:

@echo off
:: 切換當前編碼方式爲 UTF-8,處理命令行窗口標題亂碼問題
chcp 65001
title 批量壓縮當前目錄下個文件到各自壓縮包
:: 切換回默認 GBK 編碼,處理命令行輸出亂碼問題
chcp 936
echo ---------- START -------------

:: 實現一:遍歷當前目錄下(不含子目錄)的pdf文件, 壓縮成與當前文件同名的 7z 格式文件, 最後壓縮文件存放在當前目錄下新建的 7z 文件夾(可自定義,特換自己想要的文件夾名稱)下
:: for %%F in (*.pdf) do (echo "%%~nF" && "C:\Program Files\7-Zip\7z.exe" a -t7z ".\7z\%%~nF.7z" "%%F" )

:: 實現二:遍歷當前目錄下(不含子目錄)的文件夾進行壓縮, 壓縮成與當前文件夾同名的 7z 格式文件 , 最後壓縮文件存放在目錄下新建的 7z 文件夾(可自定義,特換自己想要的文件夾名稱)下(如想以目錄形式打包,可以參考此方式)
:: for /d %%F in (*) do (echo "%%~nF" && "C:\Program Files\7-Zip\7z.exe" a -t7z ".\7z\%%~nF.7z" "%%F" )

:: 實現三:遍歷當前目錄下(含子目錄)的pdf文件進行壓縮, 壓縮成與當前文件同名的 7z 格式文件 , 最後壓縮文件存放在目錄下新建的 7z 文件夾(可自定義,特換自己想要的文件夾名稱)下
:: for /r %%F in (*.pdf) do (echo "%%~nF" && "C:\Program Files\7-Zip\7z.exe" a -t7z ".\7z\%%~nF.7z" "%%F" )

:: 實現四:遍歷當前目錄下(含子目錄)的pdf文件進行壓縮, 壓縮成與當前文件同名的 7z 格式文件 , 最後壓縮文件存放在與當前文件同目錄下
:: for /r %%F in (*.pdf) do (echo "%%~nF" && "C:\Program Files\7-Zip\7z.exe" a -t7z "%%~dpnF.7z" "%%F" )

:: 實現五:遍歷當前目錄下(含子目錄)的pdf文件進行壓縮, 打包到 PDF.7z 壓縮包內 , 最後壓縮文件存放在當前目錄下
::for /r %%F in (*.pdf) do (echo "%%~nF" && 7z a -t7z "PDF.7z" "%%F" )

echo ----------  END  -------------
pause

以上展示了幾種處理方式,更多方式自行擴展,如:

  • 上述示例均已 .pdf 類型爲例,如果想壓縮其他類型文件,可以更換 *.pdf 爲對應的 *.類型 (如, *.doc ),其他通配符自行嘗試;
  • 如想將文件壓縮爲 zip 格式,則替換命令中的 -t7z ".\7z\%%~nF.7z" 替換成 -tzip ".\7z\%%~nF.zip"
  • 如想改變壓縮算法或比例等,參考下文 7z 的相關命令,進行嘗試;

使用方法

  1. 複製以上代碼內容到新建的文本文檔內;
  2. 確定要使用哪種實現來批量打包自己的文件,則刪除對應實現下 for 命令前的雙英文冒號(::,此符號等效於 rem 命令,是批處理文件的註釋標識符,雙英文冒號後面的內容是註釋說明,不會被當做代碼執行 ),並保存;
  3. 修改該新建文本文檔的擴展名 .txt 改爲 .bat,文件名自行定義;
  4. 將文件拷到要使用的目錄下,雙擊運行(切記不要管理員模式運行)。

使用上述代碼需要特別注意:

  • 當前 for 命令實現版本,不要使用管理員方式運行!不要使用管理員方式運行!不要使用管理員方式運行!重要的事情說三遍!管理員方式運行,會在 System32 目錄下生效(因管理員模式運行時,會啓用 System32 目錄下的 cmd 命令提示符來執行,而此時 for 循環則會在 ,切記!
  • 代碼中的 C:\Program Files\7-Zip\7z.exe 是本人電腦上安裝的路徑,注意修改爲自己電腦上的路徑,否則無法使用,如果將 7z 安裝路徑配置到環境變量 path 中,則可以將此處的路徑替換爲 7z.exe(或 7z

批處理 for 命令說明

for 命令可以對一組文件中的每個文件進行指定的操作,命令語法如下:

FOR %variable IN (set) DO command [command-parameters]

cmd 命令行下使用命令 for /? 可以看到 for 相關的解釋說明:

對一組文件中的每一個文件執行某個特定命令。

FOR %variable IN (set) DO command [command-parameters]

  %variable  指定一個單一字母可替換的參數。
  (set)      指定一個或一組文件。可以使用通配符。
  command    指定對每個文件執行的命令。
  command-parameters
             爲特定命令指定參數或命令行開關。

在批處理程序中使用 FOR 命令時,指定變量請使用 %%variable 而不要用 %variable。變量名稱是區分大小寫的,所以 %i 不同於 %I.

如果啓用命令擴展,則會支持下列 FOR 命令的其他格式:
	FOR /D %variable IN (set) DO command [command-parameters]

如果集中包含通配符,則指定與目錄名匹配,而不與文件名匹配。
	FOR /R [[drive:]path] %variable IN (set) DO command [command-parameters]

        檢查以 [drive:]path 爲根的目錄樹,指向每個目錄中的 FOR 語句。
        如果在 /R 後沒有指定目錄規範,則使用當前目錄。如果集僅爲一個單點(.)字符,
        則枚舉該目錄樹。

	FOR /L %variable IN (start,step,end) DO command [command-parameters]

    	該集表示以增量形式從開始到結束的一個數字序列。因此,(1,1,5)將產生序列 1 2 3 4 5,(5,-1,1)將產生序列(5 4 3 2 1)

    FOR /F ["options"] %variable IN (file-set) DO command [command-parameters]
    FOR /F ["options"] %variable IN ("string") DO command [command-parameters]
    FOR /F ["options"] %variable IN ('command') DO command [command-parameters]

        或者,如果有 usebackq 選項:

    FOR /F ["options"] %variable IN (file-set) DO command [command-parameters]
    FOR /F ["options"] %variable IN ("string") DO command [command-parameters]
    FOR /F ["options"] %variable IN ('command') DO command [command-parameters]

        fileset 爲一個或多個文件名。繼續到 fileset 中的下一個文件之前,
        每份文件都被打開、讀取並經過處理。處理包括讀取文件,將其分成一行行的文字,
        然後將每行解析成零或更多的符號。然後用已找到的符號字符串變量值調用 For 循環。
        以默認方式,/F 通過每個文件的每一行中分開的第一個空白符號。跳過空白行。
        你可通過指定可選 "options" 參數替代默認解析操作。這個帶引號的字符串包括一個
        或多個指定不同解析選項的關鍵字。這些關鍵字爲:

            eol=c           - 指一個行註釋字符的結尾(就一個)
            skip=n          - 指在文件開始時忽略的行數。
            delims=xxx      - 指分隔符集。這個替換了空格和製表符的
                              默認分隔符集。
            tokens=x,y,m-n  - 指每行的哪一個符號被傳遞到每個迭代
                              的 for 本身。這會導致額外變量名稱的分配。m-n
                              格式爲一個範圍。通過 nth 符號指定 mth。如果
                              符號字符串中的最後一個字符星號,
                              那麼額外的變量將在最後一個符號解析之後
                              分配並接受行的保留文本。
            usebackq        - 指定新語法已在下類情況中使用:
                              在作爲命令執行一個後引號的字符串並且一個單
                              引號字符爲文字字符串命令並允許在 file-set
                              中使用雙引號擴起文件名稱。

某些範例可能有助:

	FOR /F "eol=; tokens=2,3* delims=, " %i in (myfile.txt) do @echo %i %j %k

        會分析 myfile.txt 中的每一行,忽略以分號打頭的那些行,將
        每行中的第二個和第三個符號傳遞給 for 函數體,用逗號和/或
        空格分隔符號。請注意,此 for 函數體的語句引用 %i 來
        獲得第二個符號,引用 %j 來獲得第三個符號,引用 %k
        來獲得第三個符號後的所有剩餘符號。對於帶有空格的文件
        名,你需要用雙引號將文件名括起來。爲了用這種方式來使
        用雙引號,還需要使用 usebackq 選項,否則,雙引號會
        被理解成是用作定義某個要分析的字符串的。

        %i 在 for 語句中顯式聲明,%j 和 %k 是通過
        tokens= 選項隱式聲明的。可以通過 tokens= 一行
        指定最多 26 個符號,只要不試圖聲明一個高於字母 "z" 或
        "Z" 的變量。請記住,FOR 變量是單一字母、分大小寫和全局的變量;
        而且,不能同時使用超過 52 個。

        還可以在相鄰字符串上使用 FOR /F 分析邏輯,方法是,
            用單引號將括號之間的 file-set 括起來。這樣,該字符
        串會被當作一個文件中的一個單一輸入行進行解析。

最後,可以用 FOR /F 命令來分析命令的輸出。方法是,將
括號之間的 file-set 變成一個反括字符串。該字符串會
被當作命令行,傳遞到一個子 CMD.EXE,其輸出會被捕獲到
內存中,並被當作文件分析。如以下例子所示:
      FOR /F "usebackq delims==" %i IN (`set`) DO @echo %i

    	會枚舉當前環境中的環境變量名稱。


另外,FOR 變量參照的替換已被增強。你現在可以使用下列
選項語法:
     %~I          - 刪除任何引號("),擴展 %I
     %~fI        - 將 %I 擴展到一個完全合格的路徑名
     %~dI        - 僅將 %I 擴展到一個驅動器號
     %~pI        - 僅將 %I 擴展到一個路徑
     %~nI        - 僅將 %I 擴展到一個文件名
     %~xI        - 僅將 %I 擴展到一個文件擴展名
     %~sI        - 擴展的路徑只含有短名
     %~aI        - 將 %I 擴展到文件的文件屬性
     %~tI        - 將 %I 擴展到文件的日期/時間
     %~zI        - 將 %I 擴展到文件的大小
     %~$PATH:I   - 查找列在路徑環境變量的目錄,並將 %I 擴展
                   到找到的第一個完全合格的名稱。如果環境變量名
                   未被定義,或者沒有找到文件,此組合鍵會擴展到
                   空字符串

可以組合修飾符來得到多重結果:

     %~dpI       - 僅將 %I 擴展到一個驅動器號和路徑
     %~nxI       - 僅將 %I 擴展到一個文件名和擴展名
     %~fsI       - 僅將 %I 擴展到一個帶有短名的完整路徑名
     %~dp$PATH:I - 搜索列在路徑環境變量的目錄,並將 %I 擴展
                   到找到的第一個驅動器號和路徑。
     %~ftzaI     - 將 %I 擴展到類似輸出線路的 DIR

在以上例子中,%I 和 PATH 可用其他有效數值代替。%~ 語法
用一個有效的 FOR 變量名終止。選取類似 %I 的大寫變量名比較易讀,而且避免與不分大小寫的組合鍵混淆。

微軟文檔說明中語法如下

for {%%|%}<Variable> in (<Set>) do <Command> [<CommandLineOptions>]
參數 描述
{%%|%} 必需。 表示可替換參數。在命令提示符執行 for 命令時使用一個百分號 ( % ), 在命令批處理文件中執行 for 命令時使用雙百分號 ( %% ) 。變量區分大小寫,且必須使用字母表示,如 %A%B,或 %C
() 必需。 指定一個或多個文件,目錄或文本字符串,或運行該命令的一系列值。 兩邊需要圓括號。
必需。 指定要在每個文件、 目錄或文本字符串或上中包含的值的範圍縮小執行的命令設置
指定你想要指定命令中使用任何命令行選項。
/? 在命令提示符下顯示幫助。

更多介紹及示例參看此處


forfiles 命令實現

以下展示幾種通過 forfiles 命令來遍歷打包壓縮批處理文件的示例:

@echo off
:: 切換當前編碼方式爲 UTF-8,處理命令行窗口標題亂碼問題
chcp 65001
title 批量壓縮當前目錄下個文件到各自壓縮包
:: 切換回默認 GBK 編碼,處理命令行輸出亂碼問題
chcp 936
echo ---------- START -------------

::實現一:遍歷當前目錄下(含子目錄)的pdf文件進行壓縮, 壓縮成與當前文件同名的 7z 格式文件 , 最後壓縮文件存放在 7z 目錄下
:: Forfiles /P "D:\work" /s /m *.pdf /c "cmd /c echo @FNAME && 7z a -t7z D:\work\7z\@FNAME.7z @PATH"

::實現二:遍歷當前目錄下(含子目錄)的pdf文件進行壓縮, 壓縮成與當前文件同名的 7z 格式文件 , 最後壓縮文件存放在與當前文件同目錄下
:: Forfiles /P "D:\work" /s /m *.bat /c "cmd /c echo @FNAME && 7z a -t7z ./@FNAME.7z @PATH"

::實現三:遍歷當前目錄下(不含子目錄)的pdf文件進行壓縮, 壓縮成與當前文件同名的 7z 格式文件 , 最後壓縮文件存放在 7z 目錄下
:: Forfiles /P "D:\work" /m *.bat /c "cmd /c echo @FNAME && 7z a -t7z D:\work\7z\@FNAME.7z @RELPATH"

::實現四:遍歷當前目錄下(含子目錄)的pdf文件進行壓縮, 打包到 PDF.7z 壓縮包內 , 最後壓縮文件存放在當前目錄下
:: Forfiles /P "D:\work" /s /m *.bat /c "cmd /c echo @FNAME && 7z a -t7z D:\work\PDF.7z @PATH"
 
echo ----------  END  -------------
pause

以上展示了幾種 forflies 命令的處理方法,更多擴展使用,自行嘗試,使用方法類似之前 for 命令實現的使用方法。

使用上述代碼需要特別注意:

  • forfiles 命令,需要自己給定要處理的目錄路徑(如,代碼中的 D:\work 是我存放 pdf 等文件的目錄,替換成自己的目錄即可)

  • 給定目錄後,可以在任意位置執行該腳本,且可以管理員方式運行。

  • 7-zip 安裝路徑配置到環境變量 path 中,以方便在 forfilescmd 命令字符串中使用


批處理 forfiles 命令說明

forfiles 命令可以對一組文件中的每個文件進行指定的操作,命令語法如下:

FORFILES [/P pathname] [/M searchmask] [/S] 
		 [/C command] [/D [+ | -] {yyyy/MM/dd | dd}]

cmd 命令行下使用命令 forfiles /? 可以看到 forfiles 相關的解釋說明:

FORFILES [/P pathname] [/M searchmask] [/S]
         [/C command] [/D [+ | -] {yyyy/MM/dd | dd}]

描述:
    選擇一個文件(或一組文件)並在那個文件上
    執行一個命令。這有助於批處理作業。

參數列表:
    /P    pathname      表示開始搜索的路徑。默認文件夾是當前工作的
                        目錄 (.)。
    /M    searchmask    根據搜索掩碼搜索文件。默認搜索掩碼是 '*'。
    /S                  指導 forfiles 遞歸到子目錄。像 "DIR /S"。
    /C    command       表示爲每個文件執行的命令。命令字符串應該
                        用雙引號括起來。
                        
                        默認命令是 "cmd /c echo @file"。下列變量
                        可以用在命令字符串中:
                        
                        @file    - 返回文件名。
                        @fname   - 返回不帶擴展名的文件名。
                        @ext     - 只返回文件的擴展名。
                        @path    - 返回文件的完整路徑。
                        @relpath - 返回文件的相對路徑。
                        @isdir   - 如果文件類型是目錄,返回 "TRUE";
                                   如果是文件,返回 "FALSE"。
                        @fsize   - 以字節爲單位返回文件大小。
                        @fdate   - 返回文件上一次修改的日期。
                        @ftime   - 返回文件上一次修改的時間。

                        要在命令行包括特殊字符,字符請以 0xHH
                        形式使用十六進制代碼(例如,0x09 爲 tab)。
                        內部 CMD.exe 命令前面應以 "cmd /c" 開始。

    /D    date          選擇文件,其上一次修改日期大於或等於 (+),
                        或者小於或等於 (-) 用 "yyyy/MM/dd" 格式指定的日期;

                        或選擇文件,其上一次修改日期大於或等於 (+)
                        當前日期加 "dd" 天,或者小於或等於 (-) 當前

                        日期減 "dd" 天。有效的 "dd" 天數可以是
                        0 - 32768 範圍內的任何數字。如果沒有指定,

                        "+" 被當作默認符號。

    /?                  顯示此幫助消息。

示例:
    FORFILES /?
    FORFILES
    FORFILES /P C:\WINDOWS /S /M DNS*.*
    FORFILES /S /M *.txt /C "cmd /c type @file | more"
    FORFILES /P C:\ /S /M *.bat
    FORFILES /D -30 /M *.exe
             /C "cmd /c echo @path 0x09 在 30 前就被更改。"
    FORFILES /D 2001/01/01
             /C "cmd /c echo @fname 在 2001年1月1日就是新的。"
    FORFILES /D +2019/6/14 /C "cmd /c echo @fname 今天是新的。"
    FORFILES /M *.exe /D +1
    FORFILES /S /M *.doc /C "cmd /c echo @fsize"
    FORFILES /M *.txt /C "cmd /c if @isdir==FALSE notepad.exe @file"

微軟文檔說明中語法如下

forfiles [/p <Path>] [/m <SearchMask>] [/s] [/c "<Command>"] [/d [{+|-}][{<Date>|<Days>}]]
參數 描述
/p 指定開始搜索的路徑。默認情況下,搜索從當前工作目錄開始。
/m 根據指定的搜索掩碼搜索文件。默認搜索掩碼是 *.\ *
/s 指示forfiles命令以遞歸方式搜索子目錄。
在每個文件上運行指定的命令。命令字符串包含在雙引號中。默認命令是“cmd / c echo @file”。
/d [{+|-}]⁠[{|}] 選擇具有指定的時間範圍內的上次修改日期的文件。
- 選擇文件的上次修改日期晚於或等於 ( + ) 或早於或等於 ( - ) 指定的日期,其中日期採用格式 MM/DD/YYYY。
- 選擇文件的上次修改日期晚於或等於 ( + ) 的當前日期加上指定,天內或早於或等於 ( - ) 的當前日期減去天數指定。
有效值範圍 0-32,768 中包含任意數量。 如果指定沒有登錄,則 + 默認情況下使用。
/? 在命令提示符下顯示幫助。

更多介紹及示例參看此處


關於 7-zip

7-Zip 是一款擁有極高壓縮比的開源壓縮軟件。支持 Windows、Linux等平臺。通常使用 7-Zip.7z 格式能比使用 .zip 格式的壓縮檔案小 30-70%。並且使用 7-Zip 創建的 .zip 格式比大多數其它壓縮軟件創建的都小 2-10%。關於 7-zip 等更多內容,可以參看官網說明

7-zip 安裝完成後,在安裝目錄下,可以找到當前版本 7-zip.chm 幫助文檔,中文幫助文檔可以在此處下載 ,文檔下載後,打開無內容,可以嘗試在文檔屬性中勾選 解除鎖定 ,點擊應用即可。

7-zip文檔解除鎖定

7-zip 命令行的語法格式大致如下:

7z <command> [<switch>...] <base_archive_name> [<arguments>...]

7z <命令行> [<選項>...] <基本檔案名稱> [<參數變量>...]

關於 7-zip 命令行版本的更多內容,在文檔中有較爲詳細的介紹,並附有示例,可以自行學習嘗試。

7-zip 命令行版本用戶手冊


DEL 命令刪除文件

最後,附上遞歸刪除指定類型文件批處理代碼(可以用來批量刪除打包的源文件)

**特別注意:**此刪除爲完全刪除,回收站不可見,且無法撤銷還原,請慎用!!!

:: 刪除目錄下(含子目錄下)符合條件的文件
@echo off
:: 不推薦使用此方法,管理員模式運行時可能會誤刪 System32 目錄下文件,請慎用!!!
::for /r %%F in (*.7z) do (DEL /p "%%F" )

:: 推薦使用此方法,支持管理員方式運行(因提供了絕對路徑)
forfiles /P "D:\work" /s /m *.7z /c "cmd /c DEL /p @PATH"
pause

cmd 命令行下使用命令 DEL /? 可以看到 DEL 相關的解釋說明:

刪除一個或數個文件。

DEL [/P] [/F] [/S] [/Q] [/A[[:]attributes]] names
ERASE [/P] [/F] [/S] [/Q] [/A[[:]attributes]] names

  names         指定一個或多個文件或者目錄列表。
                通配符可用來刪除多個文件。
                如果指定了一個目錄,該目錄中的所
                有文件都會被刪除。

  /P            刪除每一個文件之前提示確認。
  /F            強制刪除只讀文件。
  /S            刪除所有子目錄中的指定的文件。
  /Q            安靜模式。刪除全局通配符時,不要求確認
  /A            根據屬性選擇要刪除的文件
  屬性          R  只讀文件                     S  系統文件
                H  隱藏文件                     A  存檔文件
                I  無內容索引文件               L  重分析點
                -  表示“否”的前綴

如果命令擴展被啓用,DEL 和 ERASE 更改如下:

/S 開關的顯示句法會顛倒,即只顯示已經
刪除的文件,而不顯示找不到的文件。

建議: 使用刪除時,DEL 命令後使用 /p 命令參數,來手動確認刪除文件,防止誤刪!


參考閱讀

微軟文檔 - Windows Commands

Computer Hope - Microsoft DOS and Windows command line

7-zip 中文幫助文檔

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章