當前位置:才華齋>計算機>作業系統>

Linux下批量修改檔名的方法

作業系統 閱讀(1.4W)

在Linux環境下修改檔名可以有不同的命令方式,比如rename、mv都可以進行修改,如果使用者正在瞭解這方面的知識,可以閱讀下文了解Linux修改檔名以及批量修改檔名的方法。

Linux下批量修改檔名的方法

對於單個的檔案,可以直接使用命令,那如果有大量的類似格式的檔名需要修改成其他格式的,該如何呢?

比如某次測試後,儲存的檔案為 ,,……

這一百個檔案需要在前面新增字首變成類似 ch7_,如果你還想使用mv來一個一個多痛苦啊。

當前資料夾下,檔案修改有四種方式:

1、使用while Loop加${//}來實現Lan到ch7_Lan的替換。

01find . -name "Lan*txt" -type f | read files02do03new=04mv05done複製程式碼

find . -name "Lan*txt" -type f | read filesdonew=mvdone

2、充分利用awk的分隔符功能來實現:

01ls *Lan* | awk 'BEGIN{FS="Lan";OFS="ch7_Lan"}{printf "mv "$0" ";$1=$1;print $0}' | sh複製程式碼

ls *Lan* | awk 'BEGIN{FS="Lan";OFS="ch7_Lan"}{printf "mv "$0" ";$1=$1;print $0}' | sh

首先,修改FS和OFS,輸入和輸出的分隔符。

其次,使用awk中的命令,拼接“mv ch7_”這條命令。

最後,使用|sh來執行一個shell命令來完成mv動作。

其中,$1=$1是必須的,這個是完成Lan===> ch7_Lan的關鍵(其實是可以使用$2=$2等來替換,只要一個賦值動作而已)

奇怪的是,難道沒有賦值動作,分隔符的改變就不能檢測出來麼?

3、使用awk的內建命令,gsub和system來實現替換和命令執行。

01ls "*Lan*" | awk '{org=$0;gsub("Lan", "ch7_Lan");system("mv "org" "$1)}'複製程式碼

ls "*Lan*" | awk '{org=$0;gsub("Lan", "ch7_Lan");system("mv "org" "$1)}'

首先儲存原始的資料。

其次修改 Lan為 ch_Lan

最後執行mv動作

其中,最後system命令"mv "org""需要使用""來標記。

而且org也需要"",並且不能使用$org

1、對於awk命令中的變數,不需要用$來引用。

2、在system()中,變數需要使用""來標識,否則就被當成字串來處理。

3、如果對變數進行$來進行引用,就會出現兩種情況:

3.1、正常情況下,可以被當成位置引數來引用。比如 NF表示當前行的記錄個數。比如echo "a b c d" | awk '{print NF, $NF}',這個會列印兩個變數,第一個是NF本身,其值為4;第二個是第四個位置引數,也就是$4,也就是d

3.2、非正常情況下:

3.2.1、如果變數本身就是個字串。比如echo "a b c d" | awk '{va=“varA“;print va, $va}',因為後一個引數$va,由於va是個字串,因此$va列印的就是$0

3.2.2、如果變數本身就是個數字,但是該值大於NF。 比如echo "a b c d" | awk '{va="varA";vb=6;print va, $va, vb, $vb, 1}',因為$vb相當於是$6,但是位置引數$6沒有,因此列印的就是個空字元。

01[martin@TSPerPacketsTest]$ echo "a b c d" | awk '{print NF, }'024 d03[martin@TSPerPacketsTest]$ echo "a b c d" | awk '{va="varA";print va, }'04varA a b c d05[martin@TSPerPacketsTest]$ echo "a b c d" | awk '{va="varA";vb=6;print va, , vb, , 1}'06varA a b c d 6 1複製程式碼

[martin@TSPerPacketsTest]$ echo "a b c d" | awk '{print NF, }'4 d[martin@TSPerPacketsTest]$ echo "a b c d" | awk '{va="varA";print va, }'varA a b c d[martin@TSPerPacketsTest]$ echo "a b c d" | awk '{va="varA";vb=6;print va, , vb, , 1}'varA a b c d 6 1

4、使用sed中的s來進行替換,然後使用e命令來執行:

01[martin@TSPerPacketsTest]$ ls *Lan* | sed -r -n 's/(.*)Lan(.*)/mv & 1ch_Lan2/e'複製程式碼

[martin@TSPerPacketsTest]$ ls *Lan* | sed -r -n 's/(.*)Lan(.*)/mv & 1ch_Lan2/e'

關於sed的選項,使用 -r和 -n

-r 來啟用後續可以使用 command,比如 mv

-n 來關閉 patten space中內容的顯示。

關於sed的 command,使用e。如此在使用s命令完成 pattern space中 Lan到 ch7_Lan的修改後,啟動shell來執行 pattern space中的 mv指令。

如果這裡使用 p,標記僅僅顯示 pattern space中的`指令而已。

如果對於某目錄下的所有檔案,進行檔名修改,有兩種方式:

1、彙集awk的 FS/OFS和 gsub/system來實現。有點類似於綜合上述2,3兩種方式。

01[martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk 'BEGIN{FS="/";OFS="/"}{org=$0;gsub("Lan", "ch7_Lan", );system("echo "org" "$0)}'複製程式碼

[martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk 'BEGIN{FS="/";OFS="/"}{org=$0;gsub("Lan", "ch7_Lan", );system("echo "org" "$0)}'

為了實現所有目錄下的檔名進行修改,但是又不影響路徑上資料夾名。

首先,標記FS和OFS都為"/"。

其次,使用gsub時候,指定僅僅修改當前行中最後一個記錄,使用$NF來指定。

當然,對於FS和OFS的指定,可以不放置在 BEGIN中實現,比如:

01[martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk -vF=/ -vOFS=/ '{org=$0;gsub("Lan", "ch7_Lan", );system("echo "org" "$0)}'02[martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk -F/ -vOFS=/ '{org=$0;gsub("Lan", "ch7_Lan", );system("echo "org" "$0)}'複製程式碼

[martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk -vF=/ -vOFS=/ '{org=$0;gsub("Lan", "ch7_Lan", );system("echo "org" "$0)}'[martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | awk -F/ -vOFS=/ '{org=$0;gsub("Lan", "ch7_Lan", );system("echo "org" "$0)}'

2、或者使用 sed。

01[martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | sed -r -n 's/(.*)([^/]*)Lan([^/]*)$/mv & 12ch7_Lan3/e'複製程式碼

[martin@TSPerPacketsTest]$ find . -name "*Lan*" -type f | sed -r -n 's/(.*)([^/]*)Lan([^/]*)$/mv & 12ch7_Lan3/e'

其中的各個 option和 command引數,上述都已經有介紹了。

這裡,針對資料夾名和檔名,使用了簡單的sed內建的正則表示式來進行匹配而已,來實現僅僅對檔名的修改。

以上步驟在Linux修改檔名以及批量修改檔名的實現方式,是在shell中使用find、exec、xargs、mv命令組合批量替換檔名中的字串。