Kann jemand eine sichere Lösung empfehlen, um Leerzeichen durch Unterstriche in Datei- und Verzeichnisnamen zu ersetzen, beginnend mit einem bestimmten Stammverzeichnis. Zum Beispiel:
$ tree
.
|-- a dir
| `-- file with spaces.txt
`-- b dir
|-- another file with spaces.txt
`-- yet another file with spaces.pdf
wird:
$ tree
.
|-- a_dir
| `-- file_with_spaces.txt
`-- b_dir
|-- another_file_with_spaces.txt
`-- yet_another_file_with_spaces.pdf
Verwenden Sie rename
(aka prename
), ein Perl-Skript, das möglicherweise bereits auf Ihrem System vorhanden ist. Mach es in zwei Schritten:
find -name "* *" -type d | rename 's/ /_/g' # do the directories first
find -name "* *" -type f | rename 's/ /_/g'
Basierend auf Jürgen's answer und in der Lage, mehrere Ebenen von Dateien und Verzeichnissen in einer einzigen Bindung mit der "Version 1.5 1998/12/18 16:16:31 rmb1" -Version von /usr/bin/rename
(einem Perl-Skript) zu verarbeiten:
find /tmp/ -depth -name "* *" -execdir rename 's/ /_/g' "{}" \;
Ich benutze:
for f in *\ *; do mv "$f" "${f// /_}"; done
Obwohl es nicht rekursiv ist, ist es ziemlich schnell und einfach. Ich bin mir sicher, dass jemand hier aktualisieren kann, um rekursiv zu sein.
Der Teil "$ {f ///_}" verwendet den Parametererweiterungsmechanismus von bash, um ein Muster in einem Parameter durch die angegebene Zeichenfolge zu ersetzen. Die relevante Syntax lautet "$ {parameter/pattern/string}". Siehe: https://www.gnu.org/software/bash/manual/html_node/Shell-Parameter-Expansion.html oder http://wiki.bash-hackers.org/syntax/pe .
find . -depth -name '* *' \
| while IFS= read -r f ; do mv -i "$f" "$(dirname "$f")/$(basename "$f"|tr ' ' _)" ; done
ich habe es zunächst nicht richtig verstanden, weil ich nicht an Verzeichnisse dachte.
sie können detox
von Doug Harple verwenden
detox -r <folder>
Eine find/umbenennen Lösung. umbenennen ist Teil von util-linux.
Sie müssen zuerst die Tiefe verringern, da ein Whitespace-Dateiname Teil eines Whitespace-Verzeichnisses sein kann:
find /tmp/ -depth -name "* *" -execdir rename " " "_" "{}" ";"
sie können dies verwenden:
find . -name '* *' | while read fname
do
new_fname=`echo $fname | tr " " "_"`
if [ -e $new_fname ]
then
echo "File $new_fname already exists. Not replacing $fname"
else
echo "Creating new file $new_fname to replace $fname"
mv "$fname" $new_fname
fi
done
bash 4.0
#!/bin/bash
shopt -s globstar
for file in **/*\ *
do
mv "$file" "${file// /_}"
done
Dieser macht ein bisschen mehr. Ich benutze es, um meine heruntergeladenen Torrents umzubenennen (keine Sonderzeichen (Nicht-ASCII), Leerzeichen, mehrere Punkte usw.).
#!/usr/bin/Perl
&rena(`find . -type d`);
&rena(`find . -type f`);
sub rena
{
($elems)[email protected]_;
@t=split /\n/,$elems;
for $e (@t)
{
$_=$e;
# remove ./ of find
s/^\.\///;
# non ascii transliterate
tr [\200-\377][_];
tr [\000-\40][_];
# special characters we do not want in paths
s/[ \-\,\;\?\+\'\"\!\[\]\(\)\@\#]/_/g;
# multiple dots except for extension
while (/\..*\./)
{
s/\./_/;
}
# only one _ consecutive
s/_+/_/g;
next if ($_ eq $e ) or ("./$_" eq $e);
print "$e -> $_\n";
rename ($e,$_);
}
}
Hier ist eine (ziemlich ausführliche) find -exec-Lösung, die Warnungen "Datei bereits vorhanden" in stderr schreibt:
function trspace() {
declare dir name bname dname newname replace_char
[ $# -lt 1 -o $# -gt 2 ] && { echo "usage: trspace dir char"; return 1; }
dir="${1}"
replace_char="${2:-_}"
find "${dir}" -xdev -depth -name $'*[ \t\r\n\v\f]*' -exec bash -c '
for ((i=1; i<=$#; i++)); do
name="${@:i:1}"
dname="${name%/*}"
bname="${name##*/}"
newname="${dname}/${bname//[[:space:]]/${0}}"
if [[ -e "${newname}" ]]; then
echo "Warning: file already exists: ${newname}" 1>&2
else
mv "${name}" "${newname}"
fi
done
' "${replace_char}" '{}' +
}
trspace rootdir _
Ich habe um dieses Skript herum gefunden, es kann interessant sein :)
IFS=$'\n';for f in `find .`; do file=$(echo $f | tr [:blank:] '_'); [ -e $f ] && [ ! -e $file ] && mv "$f" $file;done;unset IFS
Rekursive Version von Naidim's Answers.
find . -name "* *" | awk '{ print length, $0 }' | sort -nr -s | cut -d" " -f2- | while read f; do base=$(basename "$f"); newbase="${base// /_}"; mv "$(dirname "$f")/$(basename "$f")" "$(dirname "$f")/$newbase"; done
Hier ist eine Bash-Skriptlösung von angemessener Größe
#!/bin/bash
(
IFS=$'\n'
for y in $(ls $1)
do
mv $1/`echo $y | sed 's/ /\\ /g'` $1/`echo "$y" | sed 's/ /_/g'`
done
)
Ich mache nur eines für meinen eigenen Zweck ... Sie können es als Referenz verwenden.
#!/bin/bash
cd /vzwhome/c0cheh1/dev_source/UB_14_8
for file in *
do
echo $file
cd "/vzwhome/c0cheh1/dev_source/UB_14_8/$file/Configuration/$file"
echo "==> `pwd`"
for subfile in *\ *; do [ -d "$subfile" ] && ( mv "$subfile" "$(echo $subfile | sed -e 's/ /_/g')" ); done
ls
cd /vzwhome/c0cheh1/dev_source/UB_14_8
done
Für Dateien im Ordner "/ files"
for i in `IFS="";find /files -name *\ *`
do
echo $i
done > /tmp/list
while read line
do
mv "$line" `echo $line | sed 's/ /_/g'`
done < /tmp/list
rm /tmp/list
für diejenigen, die mit macOS Probleme haben, installieren Sie zuerst alle Tools:
brew install tree findutils rename
wenn Sie dann umbenennen müssen, machen Sie einen Alias für GNU find (gfind) als find und führen dann die Zeile von @Michel Krelin aus :,
alias find=gfind
find . -depth -name '* *' \
| while IFS= read -r f ; do mv -i "$f" "$(dirname "$f")/$(basename "$f"|tr ' ' _)" ; done
Nur dies findet Dateien im aktuellen Verzeichnis und benennt sie um. Ich habe dieses Alias.
find ./ -name "* *" -type f -d 1 | Perl -ple '$file = $_; $file =~ s/\s+/_/g; rename($_, $file);