Bash: вывод трубы «find» в «readarray»

Я пытаюсь найти файлы с помощью find и помещать эти файлы в массив Bash, чтобы я мог выполнять другие операции над ними (например, ls или grep ). Но я не могу понять, почему readarray не читает вывод вывода, поскольку он подключен к нему.

Скажем, у меня есть два файла в текущем каталоге: file1.txt и file2.txt . Таким образом, вывод find выглядит следующим образом:

 $ find . -name "file*" ./file1.txt ./file2.txt 

Поэтому я хочу "./file1.txt" это в массив, чьи два элемента являются строками "./file1.txt" и "./file2.txt" (без кавычек, очевидно).

Я пробовал это, среди нескольких других вещей:

 $ declare -a FILES $ find . -name "file*" | readarray FILES $ echo "${FILES[@]}"; echo "${#FILES[@]}" 0 

Как видно из вывода echo , мой массив пуст.

Итак, что именно я делаю неправильно здесь? Почему readarray не считывает вывод find качестве стандартного ввода и помещает эти строки в массив?

  • находить и копировать файлы, используя список входных данных и частичные пути к файлам
  • найти несколько условий с несколькими execs
  • В команде `sudo find`, как я могу убедиться, что команда` -exec` запускается как обычный пользователь?
  • Как переименовать несколько файлов, используя find
  • найти всю директорию с тем же именем и удалить все содержимое в centos
  • Как я могу запустить сценарий рекурсивно в каталоге
  • Найти рекурсивно все архивные файлы различных форматов архивов и искать их для шаблонов имен файлов
  • ls или find, чтобы исключить двоичные файлы
  • 3 Solutions collect form web for “Bash: вывод трубы «find» в «readarray»”

    При использовании конвейера bash запускает команды в подоболочках. Поэтому массив заполняется, но в подоболочке, поэтому родительская оболочка не имеет к нему доступа.

    Использовать замену процесса:

     readarray FILES < <(find) 

    Обратите внимание, что он не работает для файлов с символами новой строки в их именах.

    Правильное решение:

     unset a; declare -aa while IFS= read -r -u3 -d $'\0' file; do a+=( "$file" ) # or however you want to process each file done 3< <(find /tmp -type f -print0) 

    Это похоже на то, что подробно объясняет Greg's BashFAQ 020, и этот ответ охватывает .

    Не имеет проблем с нечетными именованными файлами (которые не содержат NUL в имени), с пробелами или новыми строками. И результат задается в массиве, что делает его полезным для дальнейшей обработки.

    readarray также может читать из stdin

     readarray FILES <<< "$(find . -name "file*")"; echo "${#FILES[@]}" 
    Linux и Unix - лучшая ОС в мире.