Команда UNIX для отображения папок с количеством файлов
Я хочу получить список папок на текущем уровне (не включая их вложенные папки) и просто распечатать имя папки и количество файлов в папке (желательно, если возможно, фильтрацию на *.jpg)
Возможно ли это в стандартной оболочке bash? ls -l
печатает обо всем, кроме файла:)
Ответы
Ответ 1
Я придумал этот:
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
Однако это намного медленнее, так как он должен читать часть файлов и в конечном итоге интерпретировать то, что они содержат (если повезет, он находит волшебный идентификатор в начале файла). -mindepth 1
не позволяет печатать .
(текущий каталог) в качестве другого каталога, который он ищет.
Ответ 2
Я нашел этот вопрос после того, как уже понял свой собственный script. Кажется, он соответствует вашим условиям и очень гибкий, поэтому я решил добавить его в качестве ответа.
<сильные > Преимущества:
- можно сгруппировать в любую глубину (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 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
Ответ 3
моя быстрее набирает из командной строки.:)
делают ли другие предложения какое-либо реальное преимущество над следующим?
find -name '*.jpg' | wc -l # recursive
find -maxdepth 1 -name '*.jpg' | wc -l # current directory only
Ответ 4
#!/bin/bash
for dir in `find . -type d | grep -v "\.$"`; do
echo $dir
ls $dir/*.jpg | wc -l
done;
Ответ 5
Вы можете сделать это без внешних команд:
for d in */; do
set -- "$d"*.jpg
printf "%s: %d\n" "${d%/}" "$#"
done
Или вы можете использовать awk (nawk или /usr/xpg 4/bin/awk в Solaris):
printf "%s\n" */*jpg |
awk -F\/ 'END {
for (d in _)
print d ":",_[d]
}
{ _[$1]++ }'