you 08:36
td 08:47
$
Команда
s
заменяет пробел и все, что следует за ним (максимально возможно, включая дополнительные пробелы) до следующего пробела на единственный пробел. Кавычки нужны.Почти такую же команду
sed
можно использовать, чтобы создать программу getname
, возвращающую имя пользователя:$ cat getname
who am i | sed 's/ .*//'
$ getname
you $
Другая команда
sed
применяется настолько часто, что мы поместили ее в командный файл с именем ind
. Эта команда вставляет пробелы до шага табуляции; она удобна для лучшего расположения текста при печати.Реализовать команду
ind
просто: достаточно установить символ табуляции в начале каждой строки:$ sed 's/^/→/' $*
Первая версия indВ этой версии символ табуляции вставляется даже в пустую строку, что не требуется. В лучшей версии используется возможность
sed
выбирать строки для модификации. Если команде предшествует шаблон, то изменяться будут только строки, соответствующие шаблону:sed '/./s/^/→/' $*
Вторая версияШаблон
/./
задает любую строку, в которой есть по крайней мере один символ, кроме символа перевода строки, поэтому команда s
не выполняется для пустых строк. Вспомним, что sed
выдает все строки, независимо от того, менялись они или нет, так что пустые строки по-прежнему выдаются.Есть еще один способ определения команды
ind
. Можно выполнять команды только для тех строк, которые не соответствуют выбираемому шаблону, предварив команду знаком восклицания '!'
. В командеsed '/^$/!s/^/→/' $*
Третья версияшаблон
/^$/
задает пустые строки (перевод строки сразу следует за ее началом), поэтому /^$/!
предписывает не выполнять команду для пустых строк.Как уже отмечалось,
sed
печатает каждую строку автоматически, независимо от того, какие операции над ней выполнялись (если только она не была удалена). Кроме того, можно использовать большинство команд редактора ed
. Поэтому легко составить программу sed
, которая напечатает, скажем, три первых строки входного потока, а затем завершится:sed 3q
Хотя
3q
не является законной командой ed
, для sed
она имеет смысл: копировать строки и завершить выполнение после третьей.От вас может потребоваться другая работа с данными, например вставка пробелов. Один из способов заключается в том, чтобы выходной поток
sed
пропустить через команду ind
, но поскольку редактор sed
допускает несколько команд, можно сделать это путем одного обращения к sed
(хотя и несколько необычного):sed 'S/^/→/
3q'
Обратите внимание, где находятся кавычки и символ перевода строки: команды должны быть на отдельных строках, но редактор игнорирует пробелы и символы табуляции в начале строки.
Представляется естественным с помощью рассмотренных выше приемов составить программу
head
, которая будет печатать несколько строк из каждого своего файла-аргумента. Но sed 3q
(или 10q
) настолько просто задать, что в этом никогда не возникало потребности. Однако мы ввели команду ind
, так как соответствующая последовательность для sed
длиннее. (В процессе работы над книгой мы заменили существовавшую программу на языке Си в 30 строк на одну строку команды ind
версии 2, приведенной выше.) Четкого критерия в отношении того, когда имеет смысл создавать отдельную программу из сложной командной строки, нет, поэтому мы предлагаем вам свое решение: поместите программу в свой каталог /bin
и посмотрите, будете ли вы ее действительно применять.Можно помещать команды редактора
sed
в файл и выполнять их, получая оттуда с помощью обращения:$ sed -f командный_файл
Вы можете обращаться к строкам, используя не только их номера. Так, команда
$ sed '/шаблон/q'
выдает из входного потока все строки до первой включительно, которые соответствуют шаблону, а команда
$ sed '/шаблон/d'
удаляет каждую строку, содержащую шаблон; удаление происходит до автоматического вывода строк, поэтому удаленные строки не учитываются.
Хотя автоматический вывод обычно удобен, иногда он мешает. Его можно отключить с помощью флага
-n
; в этом случае в выходном потоке появятся только строки, задаваемые явной командой вывода p. Например,$ sed -n '/шаблон/p'
эквивалентен команде
grep
. Условие сопоставления можно инвертировать, если завершить его символом !
, поэтому$ sed -n '/шаблон/!p'
эквивалентно команде
grep -v
. (Так же, как sed '/шаблон/d'
.)Для чего нужны две команды
sed
и grep
? В конце концов, grep
— всего лишь частный случай команды sed
. Это в какой-то степени объясняется историческими причинами: команда grep
появилась намного раньше, чем команда sed
. Но она не только уцелела, но и активно применялась. В силу специфики назначения обеих команд grep
значительно проще использовать, чем команду sed
, так как ее использование в типичных ситуациях настолько лаконично, насколько возможно (к тому же у нее есть возможности, отсутствующие у команды sed
; см., например, описание флага -b
). Но все-таки программы могут "умирать". Когда-то была программа с именем gres
, выполняющая простую подстановку, но она исчезла почти мгновенно, когда появилась команда sed
.Используя запись, такую же, как в редакторе
ed
, можно вставлять символы перевода строк с помощью команды sed
:$ sed '/$/\
> /'
Здесь добавляется символ перевода строки к каждой строке, и таким образом пустые строки вставляются во входной поток, а команда
$ sed 's/[→][→]*/\
>/g'
заменяет каждую последовательность пробелов или символов табуляции на символ перевода строки, т. е. разбивает входной поток на строки из одного слова. (Регулярное выражение
'[→]'
задает пробел или символ табуляции, '[→]*'
задает нуль или более таких символов, а весь шаблон — один или более пробелов и/или символов табуляции.)Можно также использовать пару регулярных выражений или номеров строк для задания диапазона строк, к которому будет применяться произвольная команда.
$ sed -n '20,30p'
Печать только строк с 20-й по 30-ю$ sed '1,10d'
Удаление строк с 1-й до 10-й (=tail +11)$ sed '1,/^$/cd'
Удаление всех строк до первой пустой включительно$ sed -n '/^$/,/^end/p'
Печать всех групп строк, начиная от пустой строки достроки, начинающейся с end
$ sed '$d'
Удаление последней строкиСтроки нумеруются с начала входного потока; обнуление не происходит с началом нового файла.
У команды
sed
есть существенное ограничение, которое, однако, отсутствует в редакторе ed
: в ней поддерживается относительная нумерация строк. В частности, операции +
и -
не действуют в выражениях, задающих номера строк, поэтому невозможно двигаться назад во входном потоке:$ sed '$-1d'
Недопустима обратная адресацияUnrecognized command: $-1d
$
Если строка считана, предыдущая исчезла навсегда: нет способа специфицировать предыдущую строку, а именно это требуется в команде. В принципе такой способ есть в команде