Aqui está uma versão do awk + for loop:
for filename in *; do NEWNAME=$(echo "$filename" | awk '{gsub("_"," "); print}'); mv "$filename" "$NEWNAME";done
E aqui está em ação
$ ls
$ echo "TEST" | tee {foo,bar}_{yolo,swag}_{whatever,else}.txt
TEST
$ ls
bar_swag_else.txt bar_yolo_else.txt foo_swag_else.txt foo_yolo_else.txt
bar_swag_whatever.txt bar_yolo_whatever.txt foo_swag_whatever.txt foo_yolo_whatever.txt
$ for filename in *; do NEWNAME=$(echo "$filename" | awk '{gsub("_"," "); print}'); mv "$filename" "$NEWNAME";done
$ ls
bar swag else.txt bar yolo else.txt foo swag else.txt foo yolo else.txt
bar swag whatever.txt bar yolo whatever.txt foo swag whatever.txt foo yolo whatever.txt
Para considerar novas linhas e possíveis renomeamentos de diretórios, aqui está a típica construção de loop IFS + find + read + while. Após a sugestão, execute primeiro a versão com find . -type d -print0
, porque se você começar a renomear os arquivos primeiro, e um subdiretório contiver sublinhado, o nome do arquivo não será alterado. Resumidamente, cuide dos diretórios e subdiretórios primeiro, depois cuide dos arquivos
$ ls
file_bar_swag.txt file_bar_yolo.txt file_test_swag.txt file_test_yolo.txt foo_bar_swag.txt foo_bar_yolo.txt foo_test_swag.txt foo_test_yolo.txt tester_dir
$ ls tester_dir/
ber_sweg_elze.txt ber_sweg_whut.txt ber_yelo_elze.txt ber_yelo_whut.txt fee_sweg_elze.txt fee_sweg_whut.txt fee_yelo_elze.txt fee_yelo_whut.txt
$ find . -type d -print0 | while IFS="" read -r -d "" filename ; do NEWNAME=$(echo "$filename" | awk '{gsub("_"," "); print}'); mv "$filename" "$NEWNAME";done
mv: ‘.’ and ‘./.’ are the same file
$ ls
file_bar_swag.txt file_bar_yolo.txt file_test_swag.txt file_test_yolo.txt foo_bar_swag.txt foo_bar_yolo.txt foo_test_swag.txt foo_test_yolo.txt tester dir
$ find . -type f -print0 | while IFS="" read -r -d "" filename ; do NEWNAME=$(echo "$filename" | awk '{gsub("_"," "); print}'); mv "$filename" "$NEWNAME";done
mv: ‘./.yolo’ and ‘./.yolo’ are the same file
$ ls
file bar swag.txt file bar yolo.txt file test swag.txt file test yolo.txt foo bar swag.txt foo bar yolo.txt foo test swag.txt foo test yolo.txt tester dir
Como você pode ver, há um pequeno problema - como o comando find também lista o diretório atual (simbolizado com um.), o comando criará um subproduto - um arquivo com um ponto à esquerda.
Adicionando ! -path .
quando você lida com diretórios resolve o problema
$ find . ! -path . -type d -print0 | while IFS="" read -r -d "" filename ; do NEWNAME=$(echo "$filename" | awk '{gsub("_"," "); print}'); mv "$filename" "$NEWNAME";done
$ find . -type f -print0 | while IFS="" read -r -d "" filename ; do NEWNAME=$(echo "$filename" | awk '{gsub("_"," "); print}'); mv "$filename" "$NEWNAME";done
$ ls
file bar swag.txt file bar yolo.txt file baz swag.txt file baz yolo.txt foo bar swag.txt foo bar yolo.txt foo baz swag.txt foo baz yolo.txt tester dir
$ ls "tester dir"
file bar.txt file baz.txt foo bar.txt foo baz.txt
$ ls -a
. .. file bar swag.txt file bar yolo.txt file baz swag.txt file baz yolo.txt foo bar swag.txt foo bar yolo.txt foo baz swag.txt foo baz yolo.txt tester dir