Почему `read line; объявлять $line` и `find... +`?

Ниже script работает, но какая цель имеет declare $line? Если я удалю его, это не сработает.

И какова разница между {} \; и {} + в команде find?

awk '{print "old="$1" new="$2}' list.txt |\
while IFS= read line; do
    declare $line
    find /path -name '*.ext' -exec sed -i "s/\b$old\b/$new/" {} +
done

Ответы

Ответ 1

declare устанавливает переменные: команда awk испускает содержимое формы old=foo new=bar. Запуск declare old=foo new=bar устанавливает эти две переменные.

Тем не менее, это неправильный и неряшливый способ сделать это. Вместо этого используйте read, чтобы непосредственно прочитать нужные поля из входного файла и назначить переменные (подробнее об этом в BashFAQ # 1):

while read -u 3 -r old new _; do
    find /path -name '*.ext' -exec sed -i "s/\b$old\b/$new/" {} +
done 3<list.txt

Чтобы сделать это более безопасным, можно также исключить буквенное содержимое для обработки как регулярных выражений:

requote() { sed 's/[^^]/[&]/g; s/\^/\\^/g' <<< "$1"; };
substquote() { sed 's/[&/\]/\\&/g' <<< "$1"; }
while read -u 3 -r old new _; do
    find /path -name '*.ext' -exec \
        sed -i "s/\b$(requote "$old")\b/$(substquote "$new")/" {} +
done 3<list.txt

Обратите внимание, что я не изменил использование \b, расширение, которое многие реализации sed не будут поддерживать. См. BashFAQ # 21 для альтернативных подходов к выполнению буквальных подстановок строк.


Для полноты (хотя эту не связанную тему действительно следовало задавать как отдельный вопрос), и в этом случае она могла быть закрыта как дубликат, как было предложено и ответила ранее), разрешите цитату из find man-страница:

  -exec command {} +
         This  variant  of the -exec action runs the specified command on
         the selected files, but the command line is built  by  appending
         each  selected file name at the end; the total number of invoca‐
         tions of the command will  be  much  less  than  the  number  of
         matched  files.   The command line is built in much the same way
         that xargs builds its command lines.  Only one instance of  `{}'
         is  allowed  within the command.  The command is executed in the
         starting directory.

Ответ 2

Объявлять или набирать Builtins, которые являются точными синонимами, позволяют модифицировать свойства переменных. Это очень слабая форма ввода, доступная на определенных языках программирования. Команда declare предназначена для версии 2 или более поздней версии Bash. Команда typeset также работает в сценариях Ksh.