各サブディレクトリとそのサブディレクトリの文字列に一致する行数を計算します。

各サブディレクトリとそのサブディレクトリの文字列に一致する行数を計算します。

現在のディレクトリの各サブディレクトリに、サブディレクトリの文字列とそのサブディレクトリ内のすべてのファイルに一致する行数を印刷したいと思います。

たとえば、私が持っている場合

cat /folder/a/file1.txt
test
x
x

cat /folder/a/file2.txt
x
test
x
test

cat /folder/b/c/file3.txt
x
test
x

test/folderinとすべてのサブディレクトリの発生回数を見たいです。ディレクトリごとに1行です。予想出力:

/folder: 4
/folder/a: 3
/folder/b: 1
/folder/b/c: 1

答え1

これはもう一つのトリックです。 stdoutを使用してすべてのサブディレクトリから再帰的な合計をキャプチャするため、stderrを乱用して結果を印刷します。

function countdirhelper {
  count=0
  string=$1
  for f in *
  do
    if [ -f "$f" ]
    then
      add=$(grep -c -- "$string" "$f")
    elif [ -d "$f" ]
    then
      add=$(cd "$f"; countdirhelper "$string")
    fi
    count=$((count + add))
  done
  printf "%s: %d\n" "$PWD" "$count" >&2
  printf %d "$count"
}

function countdir {
  countdirhelper "$1" > /dev/null
}

答え2

次のbashスクリプトは、./各結果にプレフィックスを追加することを除いて、入力ケースで機能するようです。

find . -type f |\
(
  while read FN
  do
    echo "`dirname $FN`"
    grep test "$FN" | wc -l
  done
) |\
(
  unset CTR
  declare -A CTR
  while read DN
  do
    read AA
    while [ "$DN" != "." ]
    do
      PN=${CTR[$DN]}
      [ "$PN" == "" ] && PN=0
      CTR[$DN]=$((PN+AA))
      DN="`dirname "$DN"`"
    done
  done
  for DN in "${!CTR[@]}"
  do
    echo "$DN: ${CTR[$DN]}"
  done
) | sort

関連情報