Команда UNIX для списка папок с количеством файлов

17

Я хочу получить список папок на текущем уровне (не включая их вложенные папки) и просто распечатать имя папки и количество файлов в папке (желательно, если возможно, фильтрацию на * .jpg)

Возможно ли это в стандартной оболочке bash? ls -l печатает обо всем, кроме количества файлов:)

    
задан DisgruntledGoat 10.03.2009 в 02:56
источник

5 ответов

21

Я придумал этот:

find -maxdepth 1 -type d | while read dir; do 
    count=$(find "$dir" -maxdepth 1 -iname \*.jpg | wc -l)
    echo "$dir ; $count"
done

Отбросьте второй -maxdepth 1 , если поиск в каталогах для файлов jpg должен быть рекурсивным, учитывая подкаталоги. Обратите внимание, что это учитывает только имя файлов. Вы можете переименовать файл, скрывая, что это jpg-изображение. Вы можете использовать команду file , чтобы сделать предположение о содержимом, вместо этого (теперь также ищет рекурсивно):

find -mindepth 1 -maxdepth 1 -type d | while read dir; do 
    count=$(find "$dir" -type f | xargs file -b --mime-type | 
            grep 'image/jpeg' | wc -l)
    echo "$dir ; $count"
done

Однако это намного медленнее, так как он должен читать часть файлов и в конечном итоге интерпретировать то, что они содержат (если повезет, он находит волшебный идентификатор в начале файла). % Co_de% не позволяет печатать -mindepth 1 (текущий каталог) в качестве другого каталога, который он ищет.     

ответ дан Johannes Schaub - litb 10.03.2009 в 03:09
источник
9

Я нашел этот вопрос после того, как уже понял свой собственный сценарий. Кажется, он соответствует вашим условиям и очень гибкий, поэтому я решил добавить его в качестве ответа.

<сильные> Преимущества:

  • можно сгруппировать в любую глубину (0 для . , 1 для поддиректориев первого уровня и т. д.)
  • печатает довольно вывод
  • no loop и только одна команда find , поэтому она немного быстрее в больших каталогах
  • все еще можно настроить на добавление настраиваемых фильтров (maxdepth, чтобы сделать его нерекурсивным, шаблон имени файла)

Необработанный код:

  find -P . -type f | rev | cut -d/ -f2- | rev | \
      cut -d/ -f1-2 | cut -d/ -f2- | sort | uniq -c

Обернуто в функцию и объяснено:

fc() {
  # Usage: fc [depth >= 0, default 1]
  # 1. List all files, not following symlinks.
  #      (Add filters like -maxdepth 1 or -iname='*.jpg' here.)
  # 2. Cut off filenames in bulk. Reverse and chop to the
  #      first / (remove filename). Reverse back.
  # 3. Cut everything after the specified depth, so that each line
  #      contains only the relevant directory path
  # 4. Cut off the preceeding '.' unless that's all there is.
  # 5. Sort and group to unique lines with count.

  find -P . -type f \
      | rev | cut -d/ -f2- | rev \
      | cut -d/ -f1-$((${1:-1}+1)) \
      | cut -d/ -f2- \
      | sort | uniq -c
}

Производит вывод следующим образом:

$ fc 0
1668 .

$ fc # depth of 1 is default
   6 .
   3 .ssh
  11 Desktop
  44 Downloads
1054 Music
 550 Pictures

Конечно, сначала с номером он может быть отправлен на sort :

$ fc | sort
   3 .ssh
   6 .
  11 Desktop
  44 Downloads
 550 Pictures
1054 Music
    
ответ дан NicoleC 28.08.2013 в 09:18
источник
8

моя быстрее набирает из командной строки. :)

делают ли другие предложения какое-либо реальное преимущество над следующим?

find -name '*.jpg' | wc -l               # recursive


find -maxdepth 1 -name '*.jpg' | wc -l   # current directory only
    
ответ дан m42 10.03.2009 в 03:25
источник
1
#!/bin/bash
for dir in 'find . -type d | grep -v "\.$"'; do
echo $dir
ls $dir/*.jpg | wc -l
done;
    
ответ дан John Ellinwood 10.03.2009 в 03:10
источник
1

Вы можете сделать это без внешних команд:

for d in */; do 
  set -- "$d"*.jpg
  printf "%s: %d\n" "${d%/}" "$#"
done

Или вы можете использовать awk ( nawk или / usr / xpg4 / bin / awk ) в Solaris ).

printf "%s\n" */*jpg |
  awk -F\/ 'END { 
    for (d in _) 
      print d ":",_[d] 
      }
  { _[]++ }'
    
ответ дан Dimitre Radoulov 10.03.2009 в 11:33
источник