Я знаю, что **/*.ext расширяется ко всем файлам во всех подкаталогах, соответствующих *.ext, но что такое расширение, включающее все такие файлы в текущем каталоге?
Что рекурсивно распространяется на все файлы в текущем каталоге?
Ответ 1
Это будет работать в Bash 4:
ls -l {,**/}*.ext
Для работы globstar с двумя звездочками необходимо установить параметр globstar (по умолчанию: on):
shopt -s globstar
От man bash:
globstar
If set, the pattern ** used in a filename expansion con‐
text will match a files and zero or more directories and
subdirectories. If the pattern is followed by a /, only
directories and subdirectories match.
Теперь мне интересно, не было ли когда-то ошибки в обработке globstar, потому что теперь, используя просто ls **/*.ext Ext, я получаю правильные результаты.
Несмотря на это, я посмотрел на анализ, который kenorb выполнил с использованием репозитория VLC, и обнаружил некоторые проблемы с этим анализом и в своем ответе, приведенном выше:
Сравнение с выводом команды find недопустимо, поскольку при указании -type f не -type f другие типы файлов (в частности, каталоги), а перечисленные команды ls скорее всего, так и есть. Кроме того, одна из перечисленных команд, ls -1 {,**/}*.* - которая, кажется, основана на моей выше, выводит только имена, которые включают точку для тех файлов, которые находятся в подкаталогах. Вопрос ОП и мой ответ содержат точку, поскольку в данном случае ищутся файлы с определенным расширением.
Однако наиболее важно то, что существует особая проблема с использованием команды ls с шаблоном globstar **. Многие дубликаты возникают, поскольку Bash расширяет шаблон для всех имен файлов (и имен каталогов) в проверяемом дереве. После расширения команда ls перечисляет каждого из них и их содержимое, если они являются каталогами.
Пример:
В нашем текущем каталоге находится подкаталог A и его содержимое:
A
└── AB
└── ABC
├── ABC1
├── ABC2
└── ABCD
└── ABCD1
В этом дереве ** расширяется до "AA/AB A/AB/ABC A/AB/ABC/ABC1 A/AB/ABC/ABC2 A/AB/ABC/ABCD A/AB/ABC/ABCD/ABCD1" (7 записей). Если вы сделаете вывод echo ** что точный результат вы получите, и каждая запись будет представлена один раз. Однако, если вы выполните ls ** она выведет список каждой из этих записей. Таким образом, по существу, это ls A затем ls A/AB и т.д., Поэтому A/AB отображается дважды. Кроме того, ls собирается отделить каждый вывод подкаталога отдельно:
...
<blank line>
directory name:
content-item
content-item
Таким образом, использование wc -l подсчитывает все те пустые строки и заголовки разделов имен каталогов, которые сбрасывают счет еще дальше.
Это еще одна причина, почему вы не должны анализировать ls.
В результате этого дальнейшего анализа я рекомендую не использовать шаблон globstar ни при каких обстоятельствах, кроме как перебирать дерево файлов следующим образом:
for entry in **
do
something "$entry"
done
В качестве окончательного сравнения я использовал исходный репозиторий Bash, который мне пригодился, и сделал это:
shopt -s globstar dotglob
diff <(echo ** | tr ' ' '\n') <(find . | sed 's|\./||' | sort)
0a1
> .
Я использовал tr чтобы заменить пробелы на новые строки, которые действительны только здесь, поскольку ни одно имя не содержит пробелов. Я использовал sed чтобы удалить ведущие ./ из каждой строки вывода из find. Я отсортировал вывод find поскольку он обычно не сортируется, а расширение глобусов в Bash уже отсортировано. Как видите, единственным выходом из diff был текущий каталог . вывод по find. Когда я сделал ls ** | wc -l ls ** | wc -l на выходе было почти вдвое больше строк.
Ответ 2
Этот wil печатает все файлы в текущем каталоге и его подкаталогах, которые заканчиваются на .ext.
find . -name '*.ext' -print
Ответ 3
Вы можете использовать: **/*.* для включения всех файлов рекурсивно (активировать: shopt -s globstar).
Ниже приведены результаты тестирования других вариантов и того, как они себя ведут.
Тестирование папки с 3472 файлами в образце папка хранилища VLC:
(Всего файлов по 3472 подсчитано по: find . -type f | wc -l)
-
ls -1 **/*.*- возвращает 3338 -
ls -1 {,**/}*.*- возвращает 3341 (как предложено Dennis) -
ls -1 {,**/}*- возвращает 8265 -
ls -1 **/*- возвращает 7817, кроме скрытых файлов (как предложено Dennis) -
ls -1 **/{.[^.],}*- возвращает 7869 (как предложено Деннис) -
ls -1 {,**/}.?*- возвращает 15855 -
ls -1 {,**/}.*- возвращает 20321
Итак, я думаю, что самый близкий способ перечислить все файлы рекурсивно - это первый пример (**/*.*) в соответствии с gniourf-gniourf comment (при условии, что файлы имеют правильные расширения или использовать конкретный), так как второй пример содержит несколько дубликатов, как показано ниже:
$ diff -u <(ls -1 {,**/}*.*) <(ls -1 **/*.*)
--- /dev/fd/63 2015-04-19 15:25:07.000000000 +0100
+++ /dev/fd/62 2015-04-19 15:25:07.000000000 +0100
@@ -1,6 +1,4 @@
COPYING.LIB
-COPYING.LIB
-Makefile.am
Makefile.am
@@ -45,7 +43,6 @@
compat/tdestroy.c
compat/vasprintf.c
configure.ac
-configure.ac
а другой генерирует еще больше дубликатов.
Чтобы включить скрытые файлы, используйте: shopt -s dotglob (отключите shopt -u dotglob). Он не рекомендуется, так как он может влиять на команды, такие как mv или rm, и вы можете случайно удалить неправильные файлы.
Ответ 4
$ find . -type f
Это будет список всех файлов в текущем каталоге. Затем вы можете сделать еще одну команду на выходе с помощью -exec
$find . -type f -exec grep "foo" {} \;
Это будет grep каждый файл из find для строки "foo".
Ответ 5
Почему бы просто не использовать расширение скобки, чтобы включить текущий каталог?
./{*,**/*}.ext
Расширение брекета происходит до расширения glob, поэтому вы можете эффективно делать то, что хотите, с более старыми версиями bash, и можете отказаться от обезьян с globstar в более новых версиях.
Кроме того, он считал, что хорошая практика в bash включает в себя ведущие ./ в ваших шаблонах glob.