ファイルがあれば、私は新しいファイル名が古いと同じになるような方法で、現在の作業ディレクトリ内のすべてのファイルの名前を変更するコマンドが必要ですが、元のファイル(例えばの行数に対応する接尾語を含むf
10を持っています行の後に名前をf_10
)に変更する必要があります。
これが私の(動作しない)試みです:
linenum=$(wc -l); find * -type f | grep -v sh | rename 's/^/ec/'*
ファイルがあれば、私は新しいファイル名が古いと同じになるような方法で、現在の作業ディレクトリ内のすべてのファイルの名前を変更するコマンドが必要ですが、元のファイル(例えばの行数に対応する接尾語を含むf
10を持っています行の後に名前をf_10
)に変更する必要があります。
これが私の(動作しない)試みです:
linenum=$(wc -l); find * -type f | grep -v sh | rename 's/^/ec/'*
回答:
どうですか:
for f in *; do mv "$f" "$f"_$(wc -l < "$f"); done
例えば:
$ wc -l *
10 file1
40 file2
100 file3
$ ls
file1_10 file2_40 file3_100
拡張機能(存在する場合)を保持する場合は、代わりにこれを使用します。
for f in *; do
ext="";
[[ $f =~ \. ]] && ext="."${f#*.};
mv "$f" "${f%%.*}"_$(wc -l < "$f")$ext;
done
_
。スクリプトについては、どのスクリプトですか?これはコマンドラインから実行する必要があり、スクリプトは必要ありません。一部のファイルとのみ一致させたい場合in *
は、たとえばに変更しますin *txt
。
wc -l < $f
grepの代わりに理解する方が簡単だと思います(そして実行する方が良いかもしれませんが、私はそれをチェックしていません)。
wc
しているため、別のアプローチを示したかったのです。でも結構です。
この1つのライナーを試すことができます:
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
rename
素晴らしい=)+1
(\.?[^\.]+)
ます。
(\.?[^\.]+)
は事実ですが、2番目のドットまでしか一致せず、連続するドットを持つファイル名やドットで終わるファイル名とは一致しないため、大丈夫ではありません。簡単な解決策ではありません。2つの代替方法があるようです。
.
文字クラス内でエスケープしたくない。
を使用して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
-name "*"
?テストの残り物?;)
以下のスクリプトは、単一のドットと拡張子(file.txt)、複数のドットと拡張子(file.1.txt)、連続したドット(file..foobar.txt)、およびファイル名のドット(file。またはファイル..)。
スクリプト
#!/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
ファイルには行がないことに注意してください。and file ..、したがって、行カウントは0です
スクリプトと提案された編集をレビューしてくれたterdonとHelioに感謝します。
チャットで@Helioを使用して開発された別のbash方法:
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
、ではなく..
)のみに一致する必要があります。拡張子がない場合、BASH_REMATCH
配列は空になります。filenameにデフォルト値を${BASH_REMATCH[1]:-$file}
使用し、拡張子をそのまま使用することで、これを利用できます。
ドットファイルを処理するfind
には、terdonと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