Мне нужна команда для переименования всех файлов в текущем рабочем каталоге таким образом, что новое имя файла будет таким же, как и у старого, но включая суффикс, соответствующий количеству строк исходных файлов (например, если файл f имеет 10 строк, тогда его следует переименовать в f_10).
Вот моя (нерабочая) попытка:
linenum=$(wc -l); find * -type f | grep -v sh | rename 's/^/ec/'*
Вы можете попробовать этот один лайнер:
find . -maxdepth 1 -type f -exec bash -c 'mv -i "$1" "$1.$(wc -l <"$1")"' _ {} \;
Здесь будут найдены все файлы в текущем рабочем каталоге (find . -maxdepth 1 -type f). Затем мы запускаем экземпляр оболочки над файлами, найденными для переименования файлов для добавления количества строк. Пример:
$ ls
bar.txt spam.txt foo.txt
$ find . -maxdepth 1 -type f -exec bash -c 'mv -i "$1" "$1.$(wc -l <"$1")"' _ {} \;
$ ls
bar.txt.12 foo.txt.24 spam.txt.7
Другой способ, который сохраняет расширение (если присутствует) с помощью rename:
for f in *; do rename -n "s/([^.]+)(\.?.*)/\$1_$(< "$f" wc -l)\$2/" "$f"; done
Если результат является ожидаемым, удалите опцию -n:
for f in *; do rename "s/([^.]+)(\.?.*)/\$1_$(< "$f" wc -l)\$2/" "$f"; done
Использование find:
find . -maxdepth 1 -type f -print0 | while read -d $'\0' f; do mv "$f" "$f"_$(grep -c . "$f"); done
Пример
% wc -l *
3 doit
5 foo
% find . -maxdepth 1 -type f -print0 | while read -d $'\0' f; do mv "$f" "$f"_$(grep -c . "$f"); done
% wc -l *
3 doit_3
5 foo_5
Просто для удовольствия и хихикает с помощью rename. Поскольку rename является инструментом Perl, который принимает произвольную строку, которая является eval'd, вы можете делать всевозможные махинации. Решение, которое, кажется, работает, следующее:
rename 's/.*/open(my $f, "<", $_);my $c=()=<$f>;$_."_".$c/e' *
В приведенном ниже сценарии рассматриваются несколько случаев: одна точка и расширение (файл.txt), несколько точек и расширений (файл.1.txt), последовательные точки (файл..foobar.txt) и точки в имени файла ( файл или файл ..).
Сценарий
#!/bin/bash
# Author: Serg Kolo
# Date: June 25,2015
# Description: script to rename files to file_numlines
# written for http://askubuntu.com/q/640430/295286
# Where are the files ?
WORKINGDIR=/home/xieerqi/substitutions
# Where do you want them to go ?
OUTPUTDIR=/home/xieerqi/substitutions/output
for file in $WORKINGDIR/* ;do
FLAG=0
EXT=$(printf "%s" "$file" | awk -F'.' '{printf "%s",$NF }' ) # extension, last field of dot-separated string
# EXT="${file##*.}" # Helio's advice is to use parameter expansion, but I dont know how to use it
if [ -z $EXT ]; then # we have a dot at the end case file. or something
# so we gotta change extension and filename
EXT=""
FILENAME=$(printf "%s" "$file" | awk -F '/' '{ print $NF}' )
# set flag for deciding how to rename
FLAG=1
else
FILENAME=$( printf "%s" "$file" | awk -F '/' -v var=$EXT '{gsub("."var,"");print $NF}' ) # filename, without path, lst in
fi
NUMLINES=$(wc -l "$file" | awk '{print $1}') # line count
if [ $FLAG -eq 0 ];then
echo "$file" renamed as "$OUTPUTDIR"/"$FILENAME"_"$NUMLINES"."$EXT"
# cp "$file" "$OUTPUTDIR"/"$FILENAME"_"$NUMLINES"."$EXT" # uncomment when necessary
else
echo "$file" renamed as "$OUTPUTDIR"/"$FILENAME"_"$NUMLINES""$EXT"
# cp "$file" "$OUTPUTDIR"/"$FILENAME"_"$NUMLINES""$EXT" # uncomment when necessary
fi
#printf "\n"
done
Сценарий
$./renamer.sh
/home/xieerqi/substitutions/file. renamed as /home/xieerqi/substitutions/output/file._0
/home/xieerqi/substitutions/file.. renamed as /home/xieerqi/substitutions/output/file.._0
/home/xieerqi/substitutions/file.1.jpg renamed as /home/xieerqi/substitutions/output/file.1_3.jpg
/home/xieerqi/substitutions/file.1.test.jpg renamed as /home/xieerqi/substitutions/output/file.1.test_3.jpg
/home/xieerqi/substitutions/file.1.test.txt renamed as /home/xieerqi/substitutions/output/file.1.test_2.txt
/home/xieerqi/substitutions/file.1.txt renamed as /home/xieerqi/substitutions/output/file.1_2.txt
/home/xieerqi/substitutions/file.2.jpg renamed as /home/xieerqi/substitutions/output/file.2_3.jpg
/home/xieerqi/substitutions/file.2.test.jpg renamed as /home/xieerqi/substitutions/output/file.2.test_3.jpg
/home/xieerqi/substitutions/file.2.test.txt renamed as /home/xieerqi/substitutions/output/file.2.test_2.txt
/home/xieerqi/substitutions/file.2.txt renamed as /home/xieerqi/substitutions/output/file.2_2.txt
/home/xieerqi/substitutions/foo..bar.txt renamed as /home/xieerqi/substitutions/output/foo..bar_4.txt
Обратите внимание, что есть нет строк в файле. и файл .., поэтому количество строк равно 0
Особая благодарность terdon и Helio за просмотр сценария и предложенные изменения
Еще один способ bash, разработанный с помощью @Helio в чате:
for file in *
do
echo "$file"
[[ -f "$file" ]] || continue
[[ $file =~ (.*)(\.[^.]+)$ ]]
cp "$file" "output/${BASH_REMATCH[1]:-$file}_$(wc -l < "$file")${BASH_REMATCH[2]}"
done
Странный выглядящий монокристаллический парень с чахлой второй головой ((.*)(\.[^.]+)$) должен соответствовать только правильным расширениям (.foo, а не [F5]). Если расширения нет, тогда массив BASH_REMATCH будет пустым. Мы можем воспользоваться этим, используя значение по умолчанию для имени файла ${BASH_REMATCH[1]:-$file} и просто используя расширение как есть.
Чтобы обрабатывать файлы точек, вы могли бы использовать find, как предложено [d1 ] @Helio в чате .
find -maxdepth 1 -type f -printf '%P\0' |
while IFS= read -r -d '' file
do
[[ $file =~ (.*)(\.[^.]+)$ ]]
cp "$file" "output/${BASH_REMATCH[1]:-$file}_$(wc -l < "$file")${BASH_REMATCH[2]}"
done