Sed (от англ. Stream EDitor) — консольная утилита для редактирования текстовых потоков. Изначально утилита была разработана для Unix, но позже была многократно клонирована и портирована на множество платформ. Ее клонирование привело к тому, что на практике можно получить совершенно неожиданные результаты при абсолютно одинаковых вызовах, так как клоны не во всем досконально следуют оригиналу, а уж тем более имеют разницу между собой. Разницы можно избежать, если использовать только правила, описанные POSIX.
В этом руководстве мы будем больше описывать реализацию GNU Sed, местами отмечая, где могут быть проблемы.
Введение[править]
Раньше текстовые файлы просматривались/обрабатывались построчно, т.е. пользователь, работая на терминале, не видел содержимого целиком, и строка за строкой делал свои изменения. С другой стороны, текст мог поступать на терминал порциями и получался тот же эффект. Во всех этих случаях поступающий текст образует поток символов (stream).
Команда Sed позволяет определить некоторые правила обработки (написать микропрограмму обработки) для каждой строки в потоке, и в этом собственно ее основная функция. Микропрограмма обработки составляется на внутреннем языке Sed, который объявляет специальные команды по типу заменить, удалить, подставить и т.п. Составляя из этих команд комбинации, можно писать очень сложные процедуры обработки, которые покрывают большую часть потребностей, встречающихся на практике. Однако, у языка Sed есть недостаток в том, что он не такой очевидный и иногда сложный, и не посвященному пользователю он будет не понятен.
К типичным задачам Sed можно отнести следующие:
- Вывод исходного файла фрагментами;
- Замена подстрок в файле по регулярным выражениям;
- Удаление/Вставка/Изменение фрагментов исходного текста файла по некоторым правилам, в частности, Sed довольно часто используют для автоматизированной правки конфигурационных файлов системы.
Более мощной альтернативой Sed является консольная утилита Awk, смысл которой такой же, но язык написания микропрограммы обработки сложнее и богаче. Однако, практика показывает, что сначала нужно попробовать решить задачу с помощью Sed, вместо того, чтобы «стрелять по воробьям из базуки».
Базовый вызов[править]
Утилита Sed в базовом виде выглядит так
sed <микропрограмма> <файл 1> <файл 2> ...<файл N>
В данном случае микропрограмма будет выполнена для каждого переданного файла. Если вы не передадите ни одного файла, то утилита перейдет в интерактивный режим и будет применять микропрограмму для всего, что напишет пользователь в консоль. Микропрограмма будет применяться каждый раз после нажатия клавиши ↵ Enter. Интерактивный режим полезен при отладке микропрограмм, если планируется использовать Sed в сценарии.
Вообще утилита Sed ждет данные в свой STDIN
поток, т.е. в командной оболочке данные для Sed можно передавать через конвейеры и/или дескрипторы:
# Передача данных через конвейер echo "Text fragment" | sed 'p' # Передача данных через STDIN (первый вариант) sed 'p' <<EOF Text fragment EOF # Передача данных через STDIN (второй вариант) sed 'p' <<< "Text fragment" # Передача данных через дескриптор (первый вариант) sed 'p' < file.txt # аналогично sed 'p' file.txt # Передача данных через дескриптор, используя Process Substitution в Bash (второй вариант) sed 'p' < <( printf "Text fragment" ) # Однако, так делать НЕЛЬЗЯ echo "Text fragment" | sed 'p' <<< "Text fragment 1" # Вызов пройдет, однако данные из конвейера будут проигнорированы
Если Sed было передано больше одного файла за один раз, то по умолчанию идет обработка их конкатенации (то есть одного большого потока). Чтобы отменить это поведение и обрабатывать файлы по отдельности, используйте опцию -s
или --separate
.
При разделении потока на фрагменты Sed по умолчанию использует символ перевода строки LF
(другими словами, файл обрабатывается в текстовом режиме). В текстах, которые сохранялись в Windows, это может приводить к неправильной обработке, потому что там окончания строк кодируются парой символов CRLF
, поэтому такие тексты нужно обрабатывать в бинарном режиме с помощью опции -b
или --binary
.
По умолчанию Sed выводит результаты в свой STDOUT
. Однако, здесь следует упомянуть одну особенность, связанную с выводом. Для этого нужно ближе познакомиться с тем, как работает Sed внутри себя.
Sed обрабатывает каждую строку в три шага:
- Сначала помещает входящую строку в буфер, называемый pattern space.
- Применяет к строке все инструкции микропрограммы. Все это делается в одном буфере pattern space.
- Печатает полученный результат из буфера в свой
STDOUT
.
Sed всегда делает последний шаг неявно. Однако некоторые инструкции сами по себе могут печатать буфер pattern space, из-за чего иногда происходит двойной вывод. Рассмотрим такой пример.
$ sed 'p' <<< "Hello World!" Hello World! Hello World!
Вы можете видеть, что входящая строка напечаталась дважды: один раз из-за автоматической печати и второй раз из-за инструкции p
. В данном примере мы хотели бы пресечь автоматическую печать, потому что печать уже делается командой p
, поэтому правильнее было бы использовать опцию -n
или --silent
:
$ sed -n 'p' <<< "Hello World!" Hello World!
Все же большинство команд не печатает буфер сами по себе, поэтому автоматическая печать для них просто необходима, например команда замены без дополнительных опций:
echo "Today is DATE" | sed 's/DATE/\"'"$(date)"'\"/' Today is "Mon Sep 5 11:33:41 MSK 2022" # Но, если команду замены вызвать с модификатором 'p', который требует печать явно, необходимо пресечь автоматическую печать $ echo "Today is DATE" | sed -n 's/DATE/\"'"$(date)"'\"/p' Today is "Mon Sep 5 11:33:41 MSK 2022"
Об автоматической печати следует помнить всегда, потому что часто Sed используется для замены фрагментов в файлах, т.е. есть риск повредить файл неправильной заменой. Некоторые реализации Sed учитывают автоматическую печать, поэтому она неявно отключается для ряда команд (например, автор встречал реализацию Sed, в которой для команды p
автоматическая печать отключалась неявно). Тем не менее, вы не должны делать такое предположение и должны составлять микропрограмму всегда с учетом существования автоматической печати.
Так как Sed по умолчанию только берет данные, то результаты обработки никак не влияют на оригинальные файлы. Если микропрограмма вносит изменения, и нужно применить изменения сразу в обрабатываемом файле, следует вызывать утилиту с опцией -i
:
$ echo "param=value" > test.txt # Заменим значение параметра прямо в файле $ sed -i 's/param=.*/param=new_value/' test.txt # Проверим $ cat test.txt param=new_value
Если у вас есть опасения повредить файл, вы всегда можете попросить Sed создать его резервную копию перед обработкой:
$ sed -i.backup 's/param=.*/param=new_value/' test.txt # В данном примере Sed создаст резервную копию для каждого переданного ему файла. В качестве имени он возьмет исходное и # припишет к нему суффикс '.backup'. $ ls test.txt test.txt.backup
Микропрограмма Sed[править]
Микропрограмма может быть передана аргументом команды, либо она может быть записана в файле и путь к этому файлу будет передаваться аргументом. Микропрограмма вообще говоря может быть пустой: хотя это лишено смысла, такой вызов пройдет.
В общем случае микропрограмма передается первым аргументом после опций. Программа для Sed может быть комплексной, т.е. может состоять более чем из одной внутренней команды языка Sed и быть записана в несколько строк. Если программа короткая и в ней нет пробелов, то допустимо ее не закавычивать. Например, следующие вызовы будут работать одинаково:
sed p test.txt sed 'p' test.txt
Но на практике микропрограмму следует всегда помещать в одинарные кавычки, так как это дисциплинирует всегда их ставить. Одинарные кавычки необходимы потому, что содержимое программы не должно интерпретироваться командной оболочкой, кроме случаев, когда в микропрограмме есть подстановки командной оболочки.
Абстрагируясь от содержимого программы, продемонстрируем варианты передачи ее утилите.
# Передача команд через опцию -e. Такой вариант используется, чтобы отключать некоторые части микропрограммы, # если вызов sed генерируется где-либо выше. # Еще такой вариант следует использовать, когда микропрограмма относительно небольшая. sed -e '<блок команд 1>' \ -e '<блок команд 2>' \ ... -e '<блок команд 3>' \ ... # или sed --expression='<блок команд 1>' \ ... # Программа может быть оформлена в многострочном виде. Чтобы показать Sed, что продолжение команды # на следующей строке, нужно в конце написать обратный слеш, например как это сделано для команды 'a' # в следующем примере. Здесь слеш необходим, потому что команда 'a' имеет аргумент, и то, что он перенесен # на другую строку, мы помечаем слешем. # Обычно в большинстве реализаций начальные пробелы в # программе игнорируются. Если вам нужны начальные пробелы, как часть строки, то следует поставить левый слеш # в начале строки, чтобы экранировать их, как это показано ниже sed ' a\ \ This line starts with a tab ' ... # В большинстве реализаций при многострочном исполнении разрешены комментарии. sed ' # Комментарий a\ \ This line starts with a tab ' ...
Если программа для Sed очень большая, то лучше записать ее в отдельном файле и передавать через опцию -f
или --file
. Пусть следующий файл содержит инструкции для Sed:
# Файл: sedprogram.txt
s/a/A/g
s/b/B/g
s/c/C/g
тогда вы можете передать его утилите, вместо того, чтобы передавать инструкции отдельной строкой
$ sed -f sedprogram.txt <<< "abc" ABC
Еще одной опцией вызова Sed является написание интерпретируемого сценария, в котором интерпретатором является сам Sed. Перепишем предыдущий пример как сценарий Sed.
#!/bin/sed -f # Файл: sedscript.sed # Для интерпретируемого сценария нужен башенг, в котором sed указывается как интерпретатор. Помните, что опция -f, если она есть, # всегда должна стоять последней, так как она требует аргумент. s/a/A/g s/b/B/g s/c/C/g
Теперь нам достаточно сделать этот сценарий исполняемым и передавать ему данные в стандартный поток ввода:
$ chmod +x sedscript.sed $ echo "abc" | ./sedscript.sed ABC
Устройство команд[править]
Если вы читаете этот учебник последовательно, то можете задаться вопросом, почему мы еще не приступили к описанию команд. Дело в том, что если вы сначала поймете логику построения инструкций в микропрограмме, вам будет проще их понимать.
В языке Sed можно выделить следующие смысловые части:
- Адресация конкретной строки. Позволяет выбирать из потока только отдельно указанные строки, либо строки, которые соответствуют регулярному выражению.
- Диапазон. Диапазон указывает Sed какие строки нужно выбирать перед обработкой. В Sed есть несколько способов указать диапазон: можно указать выбираемые строки через числа, а можно через регулярные выражения. Если диапазон никак не указан явно, то Sed обрабатывает весь поток целиком.
- Команда. Собственно сама команда, которая исполняется над диапазоном. Команды могут быть сгруппированы через блок. Внутри блока команды отделяются точкой с запятой
;
.
В общем виде микропрограмму Sed можно представить так
### Одна команда <команда> # Пример sed -n 'p' test.txt ### Одна команда с диапазоном, на который она направлена <адресация или диапазон> <команда> # Пример sed -n '1 p' test.txt # адресация первой строки sed -n '2,8 p' test.txt # числовой диапазон # Обратите внимание, что диапазон прикрепляется только к следом идущей команде. sed -n '1 p ; p' test.txt # В предыдущем примере две команды разделены ';'. Адресация относится только к первой команде 'p', но не второй. # Так как для второй печати диапазон не указан явно, то по умолчанию она будет печатать весь поток. # Микропрограмма напечатает сначала первую строку, а потом напечатает все строки фрагмента. ### Блок # Чтобы отнести адресацию или диапазон к нескольким командам, вы должны объединить их в блок через фигурные скобки. <адресация или диапазон> { <команда1> ; <команда2> ...; <командаN> } # Примеры sed -n '1 { p ; p }' test.txt # Напечатать два раза первую строку sed -n '1,3 { p ; p }' test.txt # Напечатать два раза первые три строки # Как и простые команды, блоки разделяются ;. Если перед блоком стоит адресация или диапазон, то он относится только к # следующему за ним блоку. sed -n '1,3 { p ; p } ; 1,2 { p ; p } ; {p ; p}' test.txt
Пробелы в общем случае между отдельными частями микропрограммы не нужны, так как все команды и синтаксические якоря состоят из одного символа, однако пробелы повышают удобочитаемость кода.
Способы определения диапазона[править]
Для демонстрации диапазонов мы будем использовать файл text.txt
со следующим содержимым
строка 1 строка 2 строка 3 строка 4 строка 5
Диапазон всегда определяется первым. Для примера мы будем использовать команду p
, которая просто печатает строку.
########################################### # Числовой диапазон и адресация по числам ########################################### # Простая числовая адресация. Напечатать вторую строку в потоке $ sed -n '2 p' text.txt строка 2 # Так как числовые адреса всегда состоят из цифр, то интерпретатору Sed в общем случае разделитель между ним и командой # не нужен, другими словами, пробелы можно опускать. $ sed -n '3p' text.txt # Напечатать третью строку строка 3 # Чтобы напечатать несколько строк, нужно указать первую и последнюю строку диапазона включительно через запятую. $ sed -n '2,4p' text.txt строка 2 строка 3 строка 4 # В диапазоне можно использовать специальный знак '$', который означает последняя строка потока. $ sed -n '$p' text.txt строка 5 $ sed -n '3,$p' text.txt строка 3 строка 4 строка 5
Диапазон или адрес строки можно указать регулярным выражением. Sed использует по умолчанию BRE диалект. Если использовать опцию -E
, то можно использовать ERE диалект. Разные реализации имеют вместо опции -E
опцию -r
, которая выполняет ту же задачу. Отметим, что GNU Sed поддерживает обе эти опции для совместимости. Описание грамматик BRE и ERE выходит за рамки данного руководства, однако вы можете обратиться к следующей странице.
Для следующей демонстрации мы будем использовать такой текст:
строка 1 start строка 3 строка 4 end строка 5 строка 6
################################################## # Диапазоны и адресация по регулярным выражениям ################################################## # Выбрать строки в диапазоне по регулярным выражениям. # Чтобы показать регулярное выражение, используются разделители. Подробнее мы рассмотрим эту тему позже. # Выбрать строки между start и end включительно $ sed -n '/start/,/end/p' text.txt start строка 3 строка 4 end # Данный метод можно комбинировать с числовыми диапазонами. $ sed -n '/start/,$p' text.txt start строка 3 строка 4 end строка 5 строка 6 $ sed -n '1,/end/p' text.txt строка 1 start строка 3 строка 4 end # По регулярному выражению можно выбрать конкретную строку или строки. sed -n '/строка [[:digit:]]*/p' text.txt строка 1 строка 3 строка 4 строка 5 строка 6
Команды Sed[править]
s/<шаблон>/<замена>/
(от англ. substitute)- Позволяет заменить одну подстроку по регулярному выражению на другую подстроку. Данная команда, вероятно, самая часто используемая в Sed. У данной команды есть дополнительные опции, часть из которых совпадает по назначению с некоторыми отдельными командами.
d
(от англ. delete)- Позволяет удалить строку из потока. Команда удаляет строку целиком вместе с символом перевода строки.
p
(от англ. print)- Печатает содержимое pattern space на текущий момент.
P
(от англ. print)- Печатает содержимое pattern space на текущий момент только до первого переноса строки.
q
(от англ. quit)- Останавливает обработку потока на конкретной строке. Данной команде нельзя передавать диапазон в границах.
w <файл>
(от англ. write)- Позволяет записать входящую строку в отдельный файл, указанный в аргументе. Sed может одновременно открыть до 10 файлов на запись с помощью этой команды.
r <файл>
(от англ. read)- Позволяет прочитать файл, указанный в аргументе, и печатает его содержимое в стандартный поток вывода. Если указана конкретная строка, то печать будет производиться после нее.
a <строка>
(от англ. append)- Позволяет добавить строку, указанную в аргументе, после той, что выбрана.
i <строка>
(от англ. insert)- Позволяет добавить строку, указанную в аргументе, перед той, что выбрана.
c <строка>
(от англ. change)- Заменяет выбранную строку на ту, что передана в аргументе команды.
y/<серия символов>/<серия трансформации>/
(от англ. yield)- Команда очень похожа на утилиту
tr
, т.е. позволяет задать шаблон, по которому следует трансформировать найденные символы.
- Команда очень похожа на утилиту
=
- Печатает номер обрабатываемой строки в стандартный поток вывода. Если Sed вызывается с опцией
-n
, на вывод номеров строк это никак не повлияет.
- Печатает номер обрабатываемой строки в стандартный поток вывода. Если Sed вызывается с опцией
G
(от англ. gap)- Добавляет строку из буфера hold space в pattern space. По умолчанию hold space пустой, поэтому если вы ранее не пользовались этим буфером, то эффект от этой команды обычно добавление пустой строки.
n
/N
(от англ. next)- Читает/Добавляет следующую строку относительно обрабатываемой в pattern space. Используется например, когда нужно вести чересстрочную обработку.
l
/l <длина>
(от англ. list)- Печатает содержимое pattern space в избыточной форме, т.е. с сохранением всех управляющих последовательностей. Если указать необязательный параметр, можно ограничить ширину колонки, т.е. все что не влезает в колонку по ширине будет переносится на новую строку.
Команда s[править]
Команду s
следует рассмотреть отдельно, потому что она отличается от всех прочих своим синтаксисом. Команда позволяет по регулярному выражению заменить одну подстроку во фрагменте другой подстрокой.
Ее общий синтаксис выглядит так:
s<d>шаблон<d>замена<d>модификаторы где <d> - разделитель; шаблон - регулярное выражение; замена - строка текста, которая заменит результат регулярного выражения; модификаторы - позволяют включить в замену дополнительные действия.
Разделитель – это некоторый единичный символ, который разделяет части команды. Это может быть любой символ, которого нет ни в шаблоне, ни в замене. Если этот символ есть в шаблоне или замене, то он должен быть экранирован \
. Собственно разделитель от того «плавающий», чтобы вы меньше пользовались экранированием. Традиционно в качестве разделителя используется слеш /
.
У команды есть модификаторы в виде буквы или числа, которые позволяют включить в подстановку некоторые дополнительные действия, либо управляют процессором регулярных выражений. Ниже перечислены модификаторы команды:
g
(от англ. global)- По умолчанию команда применяется только к самому первому вхождению во входящем фрагменте. Чтобы обработать все вхождения, необходимо использовать данный модификатор.
I
(от англ. ignore case)- Позволяет игнорировать регистр в шаблоне регулярного выражения.
p
(от англ. print)- Аналогичен одноименной команде
p
и позволяет напечатать результат замены. В основном используется, когда Sed работает в режиме silent.
- Аналогичен одноименной команде
w <файл>
(от англ. write)- Аналогичен одноименной команде
w
и позволяет записать результат замены в файл.
- Аналогичен одноименной команде
<число>
- Если указать конкретное число, то будет заменено конкретное вхождение, начиная с 1.
Продемонстрируем несколько вызовов команды s
. Пусть в файле text.txt
хранится такой текст.
Повседневная практика показывает, что новая модель организационной деятельности в значительной степени обуславливает создание модели развития! Практический опыт показывает, что сложившаяся структура организации напрямую зависит от новых предложений. Практический опыт показывает, что социально-экономическое развитие позволяет выполнить важнейшие задания по разработке экономической целесообразности принимаемых решений.
# Печать первого слова каждой строки. Примечание: в качестве замены мы ссылаемся на результат в первой группе регулярного выражения. $ sed 's/\([[:alnum:]]*\).*/\1/' text.txt Повседневная деятельности Практический Практический позволяет целесообразности # Затереть каждое второе слово в каждой строке $ sed 's/[[:alnum:]]*//2' text.txt Повседневная показывает, что новая модель организационной деятельности значительной степени обуславливает создание модели развития! Практический показывает, что сложившаяся структура организации напрямую зависит от новых предложений. Практический показывает, что социально-экономическое развитие позволяет важнейшие задания по разработке экономической целесообразности решений. # Затереть второе слово в последней строке $ sed '$ s/[[:alnum:]]*//2' text.txt Повседневная практика показывает, что новая модель организационной деятельности в значительной степени обуславливает создание модели развития! Практический опыт показывает, что сложившаяся структура организации напрямую зависит от новых предложений. Практический опыт показывает, что социально-экономическое развитие позволяет выполнить важнейшие задания по разработке экономической целесообразности решений.
Обратите внимание, что модификаторы w
и p
применяются только к команде замены. Чтобы Sed не запутался, модификаторы нельзя отделять от разделителя пробелами, в противном случае Sed будет думать, что это другая команда. Например сравните:
# Заменит слово WORD словосочетанием 'Hello World'. echo "WORD" | sed -n 's/WORD/Hello World/p ; p' ^ ^ # модификатор команды 's' | # | # команда 'p' Hello World # Печать модификатора Hello World # Печать команды 'p'
На раскрытый шаблон можно сослаться в замене через символ &
# Помещает любой входящий фрагмент в квадратные скобки $ sed 's/.*/[&]/' <<< "WORD" [WORD]
Разделители[править]
Разделители используются для отделения частей команд s
и y
. Иногда в шаблоне или отдельной части встречается символ, который используется в разделителе. В этом случае вам нужно просто экранировать этот символ, чтобы Sed смог определить границу части команды. Но когда таких символов много, большое количество экранирующих символов делает конструкцию нечитаемой. В этом случае разумно просто сменить разделитель.
Например сравните
# В следующем примере в заменяемом пути используется большое число слешей, как разделителей пути, # которые требуется экранировать. Обратите внимание, как сложно читать такое выражение. $ sed 's/SUBST/program\/lib\/module\/lib.so/' <<< "/usr/lib/SUBST" /usr/lib/program/lib/module/lib.so # В данном примере мы можем облегчить себе жизнь, если изменим разделитель, например на символ '|'. $ sed 's|SUBST|program/lib/module/lib.so|' <<< "/usr/lib/SUBST" /usr/lib/program/lib/module/lib.so # или так (разделитель '_') $ sed 's_SUBST_program/lib/module/lib.so_' <<< "/usr/lib/SUBST" /usr/lib/program/lib/module/lib.so # На примере команды 'y' (разделитель ':') $ sed 'y:ABCDEFGHIJKLMNOPQRSTUVWXYZ:abcdefghijklmnopqrstuvwxyz:' <<< "RED ZONE" red zone
Менять разделители можно и в адресации по регулярным выражениям:
# Разделитель запятая ','. Обратите внимание, что первый символ разделителя должен всегда экранироваться - такие правила у интерпретатора. $ sed -n '\,.*, p' <<EOF a b c EOF a b c # Еще пример (разделитель ':') $ sed -n '\:[ac]: p' <<EOF a b c EOF a c # В диапазоне # Для второй границы в диапазоне мы изменили разделитель на запятую. Это только для примера, так как смотрится это плохо. $ sed -n '1,\,stop, p' <<EOF a b stop c d e EOF a b stop # Для каждой границы в диапазоне можно задать свой разделитель. # В этом примере для левой границы используется ':', а для правой - ','. $ sed -n '\:start:,\,stop, p' <<EOF a b start c d stop e f EOF start c d stop
Помните, что менять разделитель имеет смысл, когда некоторый символ в шаблоне или подстановке вынуждает вас использовать большое число экранирующих символов.
Подстановки командной оболочки в микропрограмме[править]
Команда Sed может вызываться из сценария командной оболочки. В этом случае отдельные части микропрограммы могут формироваться в сценарии и через подстановки вставляться в микропрограмму Sed. В таких ситуациях вы должны использовать склейку и двойные кавычки, чтобы избежать большинства проблем с разбиением на слова после подстановки.
#!/bin/bash # Файл: Grep.sh # Следующий сценарий имитирует программу grep. PATTERN=$1 shift # Мы используем апостроф как разделитель, так как этот символ довольно редкий. # Мы используем склейку с двойными кавычками, чтобы подстановка правильно вставлялась. sed -n '\`'"$PATTERN"'`p' "$@"
Теперь попробуем вызвать:
$ chmod +x Grep.sh # Извлекаем из потока только цифры $ ./Grep.sh '[[:digit:]]' <<EOF a 1 b 2 c 3 4 d e f EOF 1 2 3 4
Многострочная запись микропрограммы[править]
При однострочной записи, команды должны отделяться друг от друга точкой с запятой:
sed -n 'p;p;p' <<< "test"
В многострочной записи признаком конца команды служит символ перевода строки. Таким образом, предыдущий пример можно записать так:
sed -n ' p p p' <<< "test" # Начальные пробелы как правило игнорируются, поэтому такая запись синтаксически верна sed -n ' p p p' <<< "test"
Многострочная запись используется, когда микропрограмма сложная. Обычно в таких процедурах много блоков, в том числе и вложенных. Следующая процедура демонстрирует законченный пример многострочной микропрограммы.
# Следующий вызов позволяет зачистить файл от пустых строк и строк с комментариями, начинающиеся на решетку. # Здесь диапазоны в общем то не нужны: они вставлены, чтобы усложнить пример. $ cat > test.conf <<EOF begin # Конфигурационный файл с комментариями timeout=5 threads=3 # size=312 name=proc end EOF $ sed -n ' 1,$ { /begin/,/end/ { s/#.*// s/[[:blank:]]*$// /^$/ d p } } ' test.conf begin timeout=5 threads=3 size=312 name=proc end
Инверсия группы[править]
Обычно вы применяете команды к диапазону строк прямо, но иногда удобнее записать команды в инвертированном виде относительно диапазона. Другими словами, вам иногда хочется сказать «не применяй эти команды к таким строкам, но ко всем остальным применяй».
В Sed поддерживается инвертирование группы команд с помощью символа !
, например
$ sed -n '2,4 !{p}' <<EOF > 1 > 2 > 3 > 4 > 5 > 6 > 7 > EOF 1 5 6 7 # Допустимо опускать фигурные скобки, если команда одна $ sed -n '2,4 !p' <<EOF ...
В предыдущем примере мы передаем утилите 7 строк. Микропрограмма записана с помощью инвертированной группы, которая говорит «печатай всё, кроме строк со второй по четвертую».
Практические примеры[править]
Ниже показаны некоторые практические примеры использования Sed, чтобы вы смогли проникнуться духом поточной обработки текста.
Зачистка от комментариев и пустых строк[править]
На практике иногда приходится просматривать конфигурационные файлы в сухом остатке, т.е. без комментариев и пустых строк. Реализовать такой фильтр можно с помощью команды Sed, например так:
sed -n 's/#.*//;s/[[:blank:]]*$//;/^$/ d;p' file.txt
Данная микропрограмма состоит из 4 команд:
- Первая команда
s
зачищает все комментарии. - Вторая команда
s
зачищает строки, состоящие из пробелов и символов табуляции. - Третья команда удаляет из потока строки с нулевой длиной.
- Последняя команда производит печать оставшихся строк.
Для примера попробуем посмотреть какие опции в конфигурационном файле демона sshd
в *nix системах сейчас определены. Для этого произведем такой вызов:
$ sudo sed -n 's/#.*//;s/[[:blank:]]*$//;/^$/ d;p' /etc/ssh/sshd_config Include /etc/ssh/sshd_config.d/*.conf PasswordAuthentication no ChallengeResponseAuthentication no UsePAM yes X11Forwarding yes PrintMotd no AcceptEnv LANG LC_* Subsystem sftp /usr/lib/openssh/sftp-server
Добавить пробел после каждого абзаца[править]
В текстах, в которых нет красной строки, абзацы обычно выделяются пустой строкой перед ними. В Sed предусмотрена команда G
, которая вставляет пробелы между строками.
Пусть у нас есть такой не оформленный текст:
И и задания разработке условий формировании идейные организации, нашей и модель и соображения административных показывает, заданий высшего в идейные и эксперимент высшего разработке в практика позволяет идейные рамки роль направлений дальнейших в идейные высшего проверки прогрессивного количественный значение финансовых что а соображения по подготовки сфера административных значение высшего позволяет отношении от что позволяет и место постоянный важные рамки организационной позволяет финансовых способствует важные образом реализации рост активности порядка, рамки требуют а сфера соображения анализа повседневная рост в отношении анализа представляет направлений а собой направлений и также рамки практика модель идейные занимаемых намеченных организации условий направлений нашей порядка, развития. Требуют участниками постоянный важные порядка, в особенности нашей намеченных количественный сфера позиций, новая соображения оценить нашей и особенности количественный реализация важные что задач. Активности анализа равным важную формировании активности позиций, условий обучения количественный же таким образом сфера место новая значение заданий развития. И анализа заданий условий количественный заданий активности рост задача соображения подготовки образом а а идейные направлений нас анализа деятельности отношении реализация дальнейших повседневная анализа представляет и порядка, рамки активности повседневная представляет форм дальнейших идейные занимаемых соображения нашей активности развития. От обучения и заданий условий оценить порядка, образом практика и организации, и реализация подготовки эксперимент а финансовых и административных задача рост образом направлений интересный в модель нашей направлений практика модель соображения проверки намеченных обучения нашей задача развития. Условий. Повседневная новая структура задания разработке активности постоянный подготовки активизации. Организационной выполнять соображения задания показывает, формировании позволяет идейные форм намеченных административных позиций, количественный дальнейших направлений особенности образом формировании деятельности финансовых организации, дальнейших намеченных намеченных дальнейших эксперимент идейные повседневная равным соображения образом нашей интересный соображения повседневная нашей и проверки отношении играет а активности играет способствует заданий роль высшего количественный поставленных нас нашей позволяет представляет позволяет количественный рост требуют организационной постоянный развития. По сфера задача направлений представляет от активности направлений организации, способствует соображения структура значение условий повседневная играет а условий.
Сохраним его в файл text.txt
и вызовем Sed:
sed 'G' text.txt # Результат И и задания разработке условий формировании идейные организации, нашей и модель и соображения административных показывает, заданий высшего в идейные и эксперимент высшего разработке в практика позволяет идейные рамки роль направлений дальнейших в идейные высшего проверки прогрессивного количественный значение финансовых что а соображения по подготовки сфера административных значение высшего позволяет отношении от что позволяет и место постоянный важные рамки организационной позволяет финансовых способствует важные образом реализации рост активности порядка, рамки требуют а сфера соображения анализа повседневная рост в отношении анализа представляет направлений а собой направлений и также рамки практика модель идейные занимаемых намеченных организации условий направлений нашей порядка, развития. Требуют участниками постоянный важные порядка, в особенности нашей намеченных количественный сфера позиций, новая соображения оценить нашей и особенности количественный реализация важные что задач. Активности анализа равным важную формировании активности позиций, условий обучения количественный же таким образом сфера место новая значение заданий развития. И анализа заданий условий количественный заданий активности рост задача соображения подготовки образом а а идейные направлений нас анализа деятельности отношении реализация дальнейших повседневная анализа представляет и порядка, рамки активности повседневная представляет форм дальнейших идейные занимаемых соображения нашей активности развития. От обучения и заданий условий оценить порядка, образом практика и организации, и реализация подготовки эксперимент а финансовых и административных задача рост образом направлений интересный в модель нашей направлений практика модель соображения проверки намеченных обучения нашей задача развития. Условий. Повседневная новая структура задания разработке активности постоянный подготовки активизации. Организационной выполнять соображения задания показывает, формировании позволяет идейные форм намеченных административных позиций, количественный дальнейших направлений особенности образом формировании деятельности финансовых организации, дальнейших намеченных намеченных дальнейших эксперимент идейные повседневная равным соображения образом нашей интересный соображения повседневная нашей и проверки отношении играет а активности играет способствует заданий роль высшего количественный поставленных нас нашей позволяет представляет позволяет количественный рост требуют организационной постоянный развития. По сфера задача направлений представляет от активности направлений организации, способствует соображения структура значение условий повседневная играет а условий.
Выравнивание текста по левому краю[править]
Иногда в тексте для выделения фрагмента делают отступ от левого края, например «красная строка», выделение списков, выделение цитат и прочее. Для этих целей можно использовать такой вызов Sed.
Пусть у нас есть такой текст (файл reminder.txt
):
На следующей неделе мы идем в универмаг. Список покупок: * Апельсины * Бананы * Одеяло * Зимняя резина * Сапоги P.S.: и не забудь купить батарейки типа AAA, если они попадутся на глаза.
Допустим мы хотим выделить здесь несколькими отступами сам список.
$ sed 's/\(^[*][[:space:]]\)/ \1/' reminder.txt На следующей неделе мы идем в универмаг. Список покупок: * Апельсины * Бананы * Одеяло * Зимняя резина * Сапоги P.S.: и не забудь купить батарейки типа AAA, если они попадутся на глаза.
Теперь допустим, что мы хотим выделить заголовок списка, добавив после него пустую строку. Самый простой вариант это добавить еще одну команду:
$ sed 's/\(^[*][[:space:]]\)/ \1/;/Список покупок:/G' text.txt На следующей неделе мы идем в универмаг. Список покупок: * Апельсины * Бананы * Одеяло * Зимняя резина * Сапоги P.S.: и не забудь купить батарейки типа AAA, если они попадутся на глаза.
Либо список заголовок списка можно просто подчеркнуть символами тире:
$ sed 's/\(^[*][[:space:]]\)/ \1/;/Список покупок:/a ---------------' text.txt На следующей неделе мы идем в универмаг. Список покупок: --------------- * Апельсины * Бананы * Одеяло * Зимняя резина * Сапоги P.S.: и не забудь купить батарейки типа AAA, если они попадутся на глаза.
Сколько строк в файле[править]
С помощью Sed можно узнать сколько строк в текстовом файле. Сделать это очень просто через команду =
, например:
sed -n '$=' file.txt # Пояснения: # - Команда вызывается с опцией -n, чтобы пресечь автоматический вывод. # - Мы используем $, чтобы перейти на последнюю строку. # - Командой = мы напечатаем номер последней строки.
Формирование новых списков через пересечения[править]
Sed может использоваться для порождения новых данных на основе существующих, если между ними есть связность. Например можно реализовать что-то похожее на то, как работает реляционная база данных.
Пусть у нас есть два файла, в которых хранятся таблицы с данными. Мы будем использовать договоренность, что данные начинаются с третей строки, чтобы первые две использовать под заголовки. В первой таблице мы будем хранить идентификационные номера пользователей (файл users.txt
).
ID NAME ---- -------------------------------------------------- 1001 Alice Brown 1002 Bob McDonald 5000 Tracy Presley 8544 Ron Tornton
Во второй таблице мы будем хранить контактную информацию (файл contacts.txt
).
ID Place Phone ---- --------------------- --------------- 1001 3L24 555-0124 1002 3R65 555-0265 6000 3L02 555-0002 7000 2R06 555-0106 8544 4R44 555-0044 5000 1L02 555-0102 3000 0R25 555-0025
Наша задача на основе этих таблиц получить еще одну, в которой заменить известные идентификаторы реальными именами людей. Реализовать это можно, например, такой сложной командой:
sed $(sed -n '3,$p' users.txt | awk '{print "-e s/"$1"/"$2"_"$3"/"}') <<< "$(sed -n '1p;3,$p' contacts.txt)" | column -t | tr '_' ' '
Результат
ID Place Phone Alice Brown 3L24 555-0124 Bob McDonald 3R65 555-0265 6000 3L02 555-0002 7000 2R06 555-0106 Ron Tornton 4R44 555-0044 Tracy Presley 1L02 555-0102 3000 0R25 555-0025
На самом деле мы используем целый конгломерат команд, потому что нам нужно много вспомогательных действий. Давайте разберем всю конструкцию по частям.
Команда строится из конвейера:
- Первая команда
sed
готовит новую таблицу. Она получается путем замены в первой колонке идентификаторов таблицыcontacts.txt
на известные имена сотрудников из таблицыusers.txt
, причем неизвестные идентификаторы будут сохраняться. Чтобы это сделать, мы используем команду в подоболочке. В ней мы используем Awk, чтобы подготовить микропрограмму с командами вида-e s/<ID>/<NAME>/
, причем мы ожидаем, что имя состоит из двух слов, и по этой причине мы временно вставляем вместо пробела символ_
, чтобы он не влиял на разбивку по колонкам. Awk используется только из удобства доступа к параметрам по ссылкам$1
,$2
и$3
, и в принципе можно придумать, как это можно оформить по другому. Далее мы передаем файлcontacts.txt
как поток командеsed
. - Командой
column
из пакета util-linux мы формируем таблицу, чтобы колонки красиво выравнивались по ширине. - Последняя команда
tr
используется для замены_
на простой пробел.
Препроцессоры[править]
На базе Sed можно строить простые текстовые препроцессоры, т.е. программы, которые приводят форматирование или структуру текстового файла к некоторому типовому виду. Обычно это нужно в ситуациях, когда требуется создать много файлов с однотипной структурой (например, форм для отчетов), которые затем будут переданы другой программе или ручного заполнения человеком.
Для создания однотипных файлов обычно подготавливается шаблонный файл, в котором используется специальные отметки (токены), называющие позицию, но не конкретизирующие содержимое фрагмента на этой позиции. Когда вы отдаете этот файл Sed, то он заменяет токены на то содержимое, которое вы попросите. Это могут быть элементы другого языка разметки или простой текст.
В данном примере мы попробуем написать препроцессор для генерации файла отчета. Наши отчеты будут простыми текстовыми файлами без каких-либо языков разметки. Для начала мы договоримся о токенах, которые могут встречаться в шаблонном файле.
- Заголовок. Мы будем поддерживать заголовки двух уровней. Для обозначения заголовка мы введем токен
h.<цифра>.<текст заголовка>
, где цифра означает уровень заголовка, начиная с единицы. Под цифрой 0 мы будем иметь в виду шапку документа. - Простые замены. В наш отчет будут добавляться типовая информация типа версия препроцессора, сгенерировавшего отчет; время генерации и автор. Мы будем использовать для это соответствующие токены:
DATE
,VERSION
,AUTHOR
. - Типовой фрагмент. В отчет будут добавляться типовые фрагменты текста, такие как нижний колонтитул (
FOOTER
) и информацию о лицензии (LICENSE
).
Шаблонный файл отчета пусть будет размечен так (report_templ.txt
):
h.0.report DATE VERSION AUTHOR h.1.License LICENSE h.1.Preface h.1.Section 1 h.2.Section 1.1 FOOTER
Мы будем вызывать Sed не сами по себе, а в рамках Bash-сценария, так как Sed не имеет возможности получать некоторые подстановки типа даты и автора. Для начала создадим файл с кодом Sed, которые решает простейшие задачи форматирования.
# # Файл: report.rules.sed # # Подстановка постоянных фрагментов из других файлов /LICENSE/ { # Вставляем содержимое файла с лицензией r license.txt d } /FOOTER/ { # Вставляем содержимое файла с нижним колонтитулом r footer.txt d } # Генерируем стандартное содержимое /h\.[[:digit:]]*\..*/ { /h\.0\..*/ { # Окантовка заголовка i ================================================================================ a ================================================================================ # Заголовок переводится в верхний регистр. Директива \U работает только в GNU Sed s/h\.0\.\(.*\)/ \U\1/ } /h\.[1-2]\..*/ { /h\.1\..*/ { # Окантовка заголовка первого уровня i -------------------------------------------------------------------------------- a -------------------------------------------------------------------------------- # Добавляем пустую строку a } /h\.2\..*/ { # Окантовка заголовка второго уровня a -------------------------------------------------------------------------------- a } # Заменяем текст в заголовках s/h\.[1-2]\.\(.*\)/ \1/ } } # Печатаем результаты p
Правила файла report.rules.sed
позволяют назначить стандартное форматирование для токенов заголовков, а также подставляет содержимое файла с лицензией license.txt
и файла с нижним колонтитулом footer.txt
. Чтобы заголовки выделялись в тексте, мы используем псевдографику через подчеркивание разными символами. Данные правила также преобразуют токены заголовков шаблона в нормальные строки.
Содержимое файла с лицензией license.txt
пусть будет следующим:
MIT License Copyright (c) YEAR AUTHOR1 Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
В файле с лицензией мы используем два токена YEAR
и AUTHOR1
, чтобы эти части динамически заменялись. Файл с нижним колонтитулом footer.txt
имеет такое содержимое:
================================================================================ This file was generated on TODAY at TIME
Здесь также есть заменяемые части.
Теперь оформим вызов в виде Bash скрипта.
#!/bin/bash VERSION=0.1 TEMPLATE_PATH=report_templ.txt SED_RULES='report.rules.sed' REPORT_NAME=$(mktemp -u XXXXXXXXXX.report.txt) # Подстановки DATE=$(date --rfc-3339=seconds) DATE1=$(date --rfc-3339=date) TIME=$(date +%T) YEAR=$(date +%Y) AUTHOR=$(whoami) [[ -f $TEMPLATE_PATH && -f $SED_RULES ]] || { echo "Error: Not found '$TEMPLATE_PATH' or '$SED_RULES' file"; exit 1;} echo "New report has name '$REPORT_NAME'" # Базовая процедура sed -nf "$SED_RULES" \ "$TEMPLATE_PATH" >"$REPORT_NAME" # Разрешаем подстановки sed -i \ -e 's/VERSION/Version: '"$VERSION"'/' \ -e 's/DATE/Creation date: '"$DATE"'/' \ -e 's/TODAY/'"$DATE1"'/' \ -e 's/TIME/'"$TIME"'/' \ -e 's/YEAR/'"$YEAR"'/' \ -e 's/^AUTHOR$/Author: '"$AUTHOR"'/' \ -e 's/\<AUTHOR1\>/'"$AUTHOR"'/' "$REPORT_NAME" if [[ -f $REPORT_NAME ]]; then echo "=======" echo "PREVIEW" echo "=======" cat "$REPORT_NAME" fi exit 0
Каждый вызов данного сценария генерирует новый файл с отчетом, имя для которого генерируется случайно. Для того чтобы сценарий правильно работал, все необходимые файлы должны лежать рядом с ним. В конце процедуры отобразиться превью файла, поэтому файл с отчетом можно дополнительно не открывать.
Попробуем сделать вызов.
./gen-rep.sh New report has name 'SXWtKYofTQ.report.txt' ======= PREVIEW ======= ================================================================================ REPORT ================================================================================ Creation date: 2022-09-07 14:26:05+03:00 Version: 0.1 Author: john -------------------------------------------------------------------------------- License -------------------------------------------------------------------------------- MIT License Copyright (c) 2022 john Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -------------------------------------------------------------------------------- Preface -------------------------------------------------------------------------------- -------------------------------------------------------------------------------- Section 1 -------------------------------------------------------------------------------- Section 1.1 -------------------------------------------------------------------------------- ================================================================================ This file was generated on 2022-09-07 at 14:26:05
Обратите внимание, что не все части в нашем сценарии переносятся между клонами Sed. Так, следующая конструкция поддерживается только в GNU Sed s/h\.0\.\(.*\)/ \U\1/
.
Продвинутые приёмы[править]
Работа с дополнительным буфером[править]
Ранее мы говорили, что Sed работает со входящей строкой в буфере pattern space, однако в Sed есть еще один дополнительный буфер, который позволяет временно хранить строки, который называется hold space.
Следующие команды позволяют вам получать доступ к дополнительному буферу.
h
/H
(от англ. hold)- Копирует/Добавляет содержимое из pattern space в hold space. Команда
H
добавит строку к уже имеющемуся содержимому в hold space, отделив новые данные от старых символом перевода строки.
- Копирует/Добавляет содержимое из pattern space в hold space. Команда
g
/G
(от англ. gap)- Копирует/Добавляет содержимое из hold space в pattern space, т.е. в обратную сторону.
x
(от англ. exchange)- Меняет местами содержимое pattern space и hold space.
Одним из примеров, когда можно воспользоваться буферами, это объединение подряд идущих строк попарно. Сделать это можно в реализации GNU Sed, где есть возможность относительно просто ссылаться на четные и нечетные строки. Рассмотрим следующий код:
$ printf "%s\n" a b c d | sed -n '1~2h; 2~2 { H; g; s/\n//gp }' ab cd
В этом примере мы посылаем четыре строки, каждая из которых состоит из одной буквы. Мы используем специальную адресацию, которая поддерживается в GNU Sed, позволяющая нам адресовать нечетные строки (1~2
) и четные строки (2~2
). Когда мы захватываем нечетную строку, то мы просто копируем ее в hold space. Для четных строк мы сначала добавляем ее в hold space к тому, что сейчас в нем есть, а затем копируем из hold space в pattern space. По факту мы извлекаем обратно две склеенных строки разделенные переносом строки. Последней командой мы зачищаем все переносы строки и выводим результат на экран. Так как каждая нечетная строка затирает hold space от предыдущего результата во время копирования, нам не нужно делать никаких дополнительных действий.
Еще одна техника с дополнительным буфером, это захват текста абзацами, если они отделены друг от друга пустыми строками. Пусть у нас есть такой текст:
Жил-был поп, Толоконный лоб. Пошел поп по базару Посмотреть кой-какого товару. Навстречу ему Балда Идет, сам не зная куда. «Что, батька, так рано поднялся? Чего ты взыскался?» Поп ему в ответ: «Нужен мне работник: Повар, конюх и плотник. А где найти мне такого Служителя не слишком дорогого?»
$ sed '/./{H;$!d} ; x ; s/^/\nSTART-->/ ; s/$/\n<--END/' text.txt START--> Жил-был поп, Толоконный лоб. Пошел поп по базару Посмотреть кой-какого товару. <--END START--> Навстречу ему Балда Идет, сам не зная куда. «Что, батька, так рано поднялся? Чего ты взыскался?» <--END START--> Поп ему в ответ: «Нужен мне работник: Повар, конюх и плотник. А где найти мне такого Служителя не слишком дорогого?» <--END
Главной частью здесь является /./{H;$!d} ; x ;...
. Первая команда захватывает не пустые строки до первой пустой. Такой захват происходит из-за $!d
, потому что команда d
прерывает исполнение обработки текущей строки и начинает новый цикл (Sed переходит на следующую строку и начинает микропрограмму заново), т.е. переход на следующую команду x
не произойдет до тех пор, пока Sed не упрется в пустую строку (когда d
не исполнится). Когда Sed упирается в пустую строку или конец потока, происходит переход на команду x
, которая меняет содержимое буферов местами, и мы работаем с накопленной строкой, как с одной большой строкой. В данном примере мы пометили начало и конец этой строки, чтобы показать, что абзацы захватываются целиком.
Продвинутые средства адресации в GNU Sed[править]
В реализации GNU Sed есть несколько более продвинутых методов адресации строк, нежели у других клонов.
<строка>~<шаг>
- Позволяет захватывать строки в потоке с некоторым шагом. Этим выражением мы говорим «брать каждую
<шаг>
строку, начиная со строки<строка>
». Так, выражение1~2
просит брать нечетные строки (брать через 2, начиная с первой), а2~2
— брать четные строки.
- Позволяет захватывать строки в потоке с некоторым шагом. Этим выражением мы говорим «брать каждую
<адрес>,+N
- Позволяет захватить строку, адресованную через
<адрес>
и ещеN
строк после нее.
- Позволяет захватить строку, адресованную через
<адрес>,~N
- Позволяет захватить строку, адресованную через
<адрес>
и еще несколько строк после нее до строки, кратнойN
.
- Позволяет захватить строку, адресованную через
Примеры
# Нечетные строки $ seq 8 | sed -n '1~2p' 1 3 5 7 # Каждая вторая, начиная с третьей $ seq 8 | sed -n '3~2p' 3 5 7 # Первая и еще три после нее $ seq 8 | sed -n '1,+3p' 1 2 3 4 # Строка с цифрой 5 и еще три после нее $ seq 8 | sed -n '/5/,+3p' 5 6 7 8 # С первой и до строки, чей номер кратен 3 $ seq 8 | sed -n '1,~3p' 1 2 3
Переходы в микропрограмме в GNU Sed[править]
В этом разделе мы попробуем прояснить, как применяется микропрограмма к обрабатываемому тексту. Начнем с того, что применение микропрограммы ко входящей строке принято называть циклом. Микропрограмма в цикле применяется к строке с первой по последнюю команду прежде чем перейти к следующей строке и начать новый цикл. Однако, некоторые команды способны прервать цикл, либо перейти в другую его точку. Переходы позволяют реализовать что-то похожее на ветвления, чтобы при определенных условиях выполнять одни команды, а при других условиях — другие.
Следующие команды способны влиять на цикл:
q
- Безусловно прерывает текущий цикл и завершает обработку всего потока.
d
- Очищает pattern space и начинает новый цикл.
D
- Если pattern space пуст, то работает как
d
, иначе удаляет содержимое pattern space до первой новой строки в нем, перезапускает цикл с результатом, не читая новую строку.
- Если pattern space пуст, то работает как
t <метка>
- Команда, очень похожая на оператор
goto
в некоторых языках программирования. Делает переход на метку цикла в случае успешного исполнения командыs
, чтения очередной строки, либо успешного другого перехода. Если метка отсутствует, то переход выполняется в конец цикла. Данной командой программируются условные переходы.
- Команда, очень похожая на оператор
T <метка>
- Аналогична команде
t
, но все условия перехода инвертированы, т.е. командаs
выполнилась не успешно, очередная строка не прочиталась.
- Аналогична команде
b <метка>
- Безусловный переход на указанную метку. Если метка опущена, то переход происходит в конец цикла.
Следующий пример показывает как работает D
.
# Примечание: команда l позволяет увидеть содержимое pattern space таким, какое оно есть, # вместе со всеми управляющими последовательностями. # $ printf '%s\n' aa bb cc dd | sed -n 'N;l;D' aa\nbb$ bb\ncc$ cc\ndd$
В данном примере цикл состоит из 3 команд. В первом цикле команда N
, после того как была помещена очередная строка в pattern space, добавляет к ней следующую за ней строку. Далее командой l
мы выводим содержимое pattern space, и вы можете видеть, что в буфере находится конкатенация первых двух строк, разделенная переносом строки. Далее команда D
, начиная с начала буфера, удаляет из pattern space строку до первого переноса строки, т.е. на момент конца цикла у нас в pattern space останется только bb
. Затем команда D
прерывает цикл, и Sed переходит на новую строку, но без помещения следующей строки в pattern space. Снова команда N
добавляет к имеющемуся содержимому следующую строку (bb\ncc$
) и последовательность действий повторяется.
Вы можете создать метку в любой части цикла. Для этого используется следующий синтаксис: :имя_метки
. Имя метки должно состоять по меньшей мере из одного символа. Метки не являются командами, поэтому они пропускаются при нормальном течении цикла.
Для уверенного использования данной техники необходима некоторая практика. Следующий пример демонстрирует, как работают переходы в цикле.
$ printf '%s\n' a1 a2 a3 | sed '/1/b else ; s/a/z/ ; :else ; y/123/456/' a4 z5 z6 # Предыдущая микропрограмма эквивалентна следующей $ printf '%s\n' a1 a2 a3 | sed -E '/1/!s/a/z/ ; y/123/456/' a4 z5 z6
В этом примере демонстрируется, как строятся конструкции, чем то похожие на if-else, в Sed. Здесь команда /1/b else
играет роль условия. Если в очередной строке есть цифра 1, то исполнение цикла перейдет на метку :else
, благодаря команде b
. За меткой :else
исполняется команда y
. Именно из-за перехода мы видим, что в первой строке (где есть цифра 1) изменилась только цифра, но не буква, так как мы перепрыгнули команду, которая изменила бы букву. В оставшихся строках нет цифры 1, поэтому перехода не происходит и на строку применяется две команды, первая из которых изменит букву, а вторая цифру.
Несмотря на всю простоту, даже в Sed можно случайно создавать бесконечные петли. Следующий код демонстрирует бесконечную петлю в Sed, из-за чего исполнение программы никогда не прервется:
$ echo "looping" | sed ':begin ; b begin' # Когда исполнение микропрограммы доходит до конца, команда b отправляет точку следования в начало и так до бесконечности.
Однако, некоторые команды способны самостоятельно разорвать такую петлю из-за особенностей их работы. Например, команды n
и N
завершают исполнение Sed, когда им нечего читать в потоке:
$ seq 3 | sed ':begin ; n ; bbegin' 1 2 3 $ seq 3 | sed ':begin ; N ; bbegin' 1 2 3 # В обоих случаях конец потока когда-нибудь настанет, и команда n/N прервет исполнение петли.
На практике петли могут использоваться для накапливания информации в буфере pattern space. В официальной документации к GNU Sed приводится пример использования показанной выше петли, в котором она используется для объединения строк. Пусть у нас есть такой текст, использующий QP кодирование.
All the wor= ld's a stag= e, And all the= men and wo= men merely = players: They have t= heir exits = and their e= ntrances; And one man= in his tim= e plays man= y parts.
Не вдаваясь в подробности этого кодирования, лишь отметим, что каждая строка завершается символом =
, который говорит, что продолжение перенесено. Нашей задачей является восстановить оригинальный текст через слияние фрагментов по завершающему символу =
. В документации приводится такое решение:
$ sed ':x ; /=$/ { N ; s/=\n//g ; bx }' text.txt All the world's a stage, And all the men and women merely players: They have their exits and their entrances; And one man in his time plays many parts.
В данном примере петли порождаются только для строк, оканчивающихся знаком =
. В таких петлях буфер накапливается строками, содержащими знак равно в конце командой N
, а разрыв происходит из-за первых строк, в которых этого символа нет в конце, так как команда перехода в начало будет пропускаться. Печать окончательного результата происходит за счет автоматической печати.
К решению этой задачи есть еще один подход, основанный на условном переходе:
$ sed ':x ; $!N ; s/=\n// ; tx ; P ; D' text.txt All the world's a stage, And all the men and women merely players: They have their exits and their entrances; And one man in his time plays many parts.
Это решение похоже на предыдущее, но петля здесь строится из условного перехода t
: команда $!N
накапливает данные в буфере, а s/=\n//
удаляет пару =\n
; когда команда s
выполняется успешно, мы переходим в начало программы (т.е. еще есть что накапливать), иначе продолжения нет и мы идем дальше. Командой P
мы выводим накопленный результат, а командой D
мы начинаем новый цикл, если в потоке есть еще данные. В этом примере команда D
также пресекает автоматическую печать.
Ссылки[править]
- Официальная документация GNU Sed
Время на прочтение
7 мин
Количество просмотров 103K
Компиляция из двух смежных статей на тему использования команды sed
для редактирования текстовых файлов, включая различные варианты поиска и замены шаблонов, а также всевозможные операции со строками. Идею к публикации этого гайда подал участник nronnie в комментарии к предыдущей статье, посвящённой работе с Bash.
SED – это потоковый редактор текста (от stream editor), c помощью которого можно выполнять с файлами множество операций вроде поиска и замены, вставки или удаления. При этом чаще всего он используется именно для поиска и замены.
SED позволяет редактировать файлы, не открывая, что существенно ускоряет работу, чем при использовании того же vi. Помимо этого, SED поддерживает регулярные выражения, с помощью которых можно выполнять сложное сопоставление шаблонов.
Общий синтаксис команды sed
выглядит так:
sed OPTIONS... [SCRIPT] [INPUTFILE...]
Сами же варианты её использования мы рассмотрим на примере следующего отрывка текста:
$cat > geekfile.txt
unix is great os. unix is opensource. unix is free os.
learn operating system.
unix linux which one you choose.
unix is easy to learn.unix is a multiuser os.Learn unix .unix is a powerful.
Примеры команд
1. Замена заданного шаблона
Следующая команда заменит в целевом файле вхождения слова unix
на linux
:
$sed 's/unix/linux/' geekfile.txt
Вывод:
linux is great os. unix is opensource. unix is free os.
learn operating system.
linux linux which one you choose.
linux is easy to learn.unix is a multiuser os.Learn unix .unix is a powerful.
Здесь s
обозначает операцию замены, а прямые слэши выступают разделителями. В качестве искомого шаблона используется unix
, а в качестве его замены linux
.
По умолчанию команда sed
заменяет первое вхождение шаблона в каждой строке, не затрагивая второе, третье и т.д.
2. Замена n-ого вхождения шаблона в строке
Вхождения, которые нужно заменить, обозначаются с помощью флагов /1
, /2
и т.д. К примеру, следующая команда заменит в каждой строке второе вхождение:
$sed 's/unix/linux/2' geekfile.txt
Вывод:
unix is great os. linux is opensource. unix is free os.
learn operating system.
unix linux which one you choose.
unix is easy to learn.linux is a multiuser os.Learn unix .unix is a powerful.
3. Замена всех вхождений шаблона в файле
Дя замены всех без исключения вхождений заданного шаблона используется глобальный флаг /g
:
$sed 's/unix/linux/g' geekfile.txt
Вывод:
linux is great os. linux is opensource. linux is free os.
learn operating system.
linux linux which one you choose.
linux is easy to learn.linux is a multiuser os.Learn linux .linux is a powerful.
4. Замена всех вхождений шаблона в строке начиная с n-ого
Для этого номер вхождения, с которого нужно начать, сопровождается флагом g
. Следующая команда заменит в каждой строке третье, четвёртое и т.д. вхождения слова unix
на слово linux
:
$sed 's/unix/linux/3g' geekfile.txt
Вывод:
unix is great os. unix is opensource. linux is free os.
learn operating system.
unix linux which one you choose.
unix is easy to learn.unix is a multiuser os.Learn linux .linux is a powerful.
5. Заключение первых символов слов в скобки
Следующая команда заключит в скобки первый символ каждого слова:
$ echo "Welcome To The Geek Stuff" | sed 's/\(\b[A-Z]\)/\(\1\)/g'
Вывод:
(W)elcome (T)o (T)he (G)eek (S)tuff
6. Замена шаблона в конкретной строке
Можно ограничить выполнение команды sed
нужной строкой:
$sed '3 s/unix/linux/' geekfile.txt
Вывод:
unix is great os. unix is opensource. unix is free os.
learn operating system.
linux linux which one you choose.
unix is easy to learn.unix is a multiuser os.Learn unix .unix is a powerful.
Вышеприведённая команда заменяет заданное слово только в третьей строке.
7. Дублирование изменяемых строк
При добавлении флага /p
команда выведет в терминал строки, где производится замена. Строки, в которых искомый шаблон отсутствует, не дублируются.
$sed 's/unix/linux/p' geekfile.txt
Вывод:
linux is great os. unix is opensource. unix is free os.
linux is great os. unix is opensource. unix is free os.
learn operating system.
linux linux which one you choose.
linux linux which one you choose.
linux is easy to learn.unix is a multiuser os.Learn unix .unix is a powerful.
linux is easy to learn.unix is a multiuser os.Learn unix .unix is a powerful.
8. Вывод только строк с заменой
Если рядом с флагом /p
добавить ключ -n
, в терминале отобразятся только строки, где выполнялась замена. В данном случае -n
отключает дублирующее поведение флага /p
, поэтому строки с заменой выводятся по одному разу.
$sed -n 's/unix/linux/p' geekfile.txt
Вывод:
linux is great os. unix is opensource. unix is free os.
linux linux which one you choose.
linux is easy to learn.unix is a multiuser os.Learn unix .unix is a powerful.
Если использовать только -n
, исключив /p
, вывод команда не произведёт.
9. Замена шаблона в указанном диапазоне строк
sed
позволяет указывать диапазон строк, в которых требуется заменить определённый шаблон:
$sed '1,3 s/unix/linux/' geekfile.txt
Вывод:
linux is great os. unix is opensource. unix is free os.
learn operating system.
linux linux which one you choose.
unix is easy to learn.unix is a multiuser os.Learn unix .unix is a powerful.
Здесь sed
производит замену в строках с 1 по 3. А вот ещё один пример:
$sed '2,$ s/unix/linux/' geekfile.txt
Вывод:
unix is great os. unix is opensource. unix is free os.
learn operating system.
linux linux which one you choose.
linux is easy to learn.unix is a multiuser os.Learn unix .unix is a powerful
Здесь $
указывает на последнюю строку файла, в связи с чем sed
заменяет первые вхождения шаблона, начиная со второй и заканчивая ей.
10. Удаление строк из файла
С помощью sed
также можно удалять строки из заданного файла:
10.1 Удаление n-ой строки:
Синтаксис: $ sed 'nd' filename.txt
Пример: $ sed '5d' filename.txt
10.2 Удаление последней строки:
Синтаксис: $ sed '$d' filename.txt
10.3 Удаление строк с x> по y:
Синтаксис: $ sed 'x,yd' filename.txt
Пример: $ sed '3,6d' filename.txt
10.4 Удаление строк с n-ой до последней:
Синтаксис: $ sed 'n,$d' filename.txt
Пример: $ sed '12,$d' filename.txt
10.5 Удаление текста, соответствующего шаблону:
Синтаксис: $ sed '/pattern/d' filename.txt
Пример: $ sed '/abc/d' filename.txt
Примеры команд (продолжение)
Во второй части мы разберём ещё одну серию операций с командой sed
на примере уже другого файла, a.txt. Чтобы излишне не растягивать статью, вывод команд далее приводится не будет, так что для лучшего понимания можете самостоятельно попрактиковаться, создав такой же файл с аналогичным содержимым.
[root@rhel7 ~]# cat a.txt
life isn't meant to be easy, life is meant to be lived.
Try to learn & understand something new everyday in life.
Respect everyone & most important love everyone.
Don’t hesitate to ask for love & don’t hesitate to show love too.
Life is too short to be shy.
In life, experience will help you differentiating right from wrong.
Добавление/удаление пустых строк в файле
1. Вставить после каждой текстовой строки одну пустую:
[root@rhel7 ~]# sed G a.txt
2. Вставить две пустые строки:
[root@rhel7 ~]# sed 'G;G' a.txt
3. Удалить все пустые строки и вставить по одной после каждой текстовой:
[root@rhel7 ~]# sed '/^$/d;G' a.txt
4. Вставить пустую строку над каждой, содержащей love
:
[root@rhel7 ~]# sed '/love/{x;p;x;}' a.txt
5. Вставить пустую строку после каждой, содержащей love
:
[root@rhel7 ~]# sed '/love/G' a.txt
6. Вставить 5 пробелов слева от каждой строки:
[root@rhel7 ~]# sed 's/^/ /' a.txt
Нумерация строк
1. Пронумеровать каждую строку файла (с левым выравниванием).
В этой команде символ =
используется для нумерации строки, а флаг \t
для табулирования между номером и предложением:
Пример: [root@rhel7 ~]# sed = a.txt | sed 'N;s/\n/\t/'
2. Пронумеровать каждую строку файла (число слева, выравнивание по правому краю).
Эта команда похожа на `cat -n filename`
:
Пример: [root@rhel7 ~]# sed = a.txt | sed 'N; s/^/ /; s/ *\(.\{4,\}\)\n/\1 /'
3. Пронумеровать каждую непустую строку файла:
Пример: [root@rhel7 ~]# sed '/./=' a.txt | sed '/./N; s/\n/ /'
Удаление строк
1. Удалить конкретную строку:
Синтаксис: sed ‘nd’ filename
Пример: [root@rhel7 ~]# sed '5d' a.txt
2. Удалить последнюю строку:
Синтаксис: sed ‘$d’ filename
3. Удалить строки с x
по y
:
Синтаксис: sed ‘x,yd’ filename
Пример: [root@rhel7 ~]# sed '3,5d' a.txt
4. Удалить строки с n-ой до последней:
Синтаксис: sed ‘n,$d’ filename
Пример: [root@rhel7 ~]# sed '2,$d' a.txt
5. Удалить строку, содержащую шаблон:
Синтаксис: sed ‘/pattern/d’ filename
Пример: [root@rhel7 ~]# sed '/life/d' a.txt
6. Удалить каждую вторую строку начиная с n-ой:
Синтаксис: sed ‘n~2d’ filename
Пример: [root@rhel7 ~]# sed '3~2d' a.txt
7. Удалить строки, содержащие шаблон, и по две строки после них:
Синтаксис: sed ‘/pattern/,+2d’ filename
Пример: [root@rhel7 ~]# sed '/easy/,+2d' a.txt
8. Удалить пустые строки:
Пример: [root@rhel7 ~]# sed '/^$/d' a.txt
9. Удалить пустые строки или начинающиеся с #
:
Пример: [root@rhel7 ~]# sed -i '/^#/d;/^$/d' a.txt
Просмотр/вывод строк
Для просмотра содержимого файла мы используем команду cat
, а его начало и конец просматриваем с помощью утилит head
и tail
. Но что, если нас интересует некий участок в середине файла? В таком случае можно снова задействовать sed
.
1. Просмотреть файл со строки x по y:
Синтаксис: sed -n ‘x,yp’ filename
Пример: [root@rhel7 ~]# sed -n '2,5p' a.txt
2. Просмотреть весь файл, за исключением заданного диапазона:
Синтаксис: sed ‘x,yd’ filename
Пример: [root@rhel7 ~]# sed '2,4d' a.txt
3. Вывести n-ую строку файла:
Синтаксис: sed -n ‘address’p filename
Пример: [root@rhel7 ~]# sed -n '4'p a.txt
4. Вывести строки с x по y:
Синтаксис: sed -n ‘x,y’p filename
Пример: [root@rhel7 ~]# sed -n '4,6'p a.txt
5. Вывести только последнюю строку:
Синтаксис: sed -n ‘$’p filename
6. Вывести с n-ой строки до последней:
Синтаксис: sed -n ‘n,$p’ filename
Пример: [root@rhel7 ~]# sed -n '3,$'p a.txt
7. Вывести строки, содержащие указанный шаблон:
Синтаксис: sed -n /pattern/p filename
Пример: [root@rhel7 ~]# sed -n /every/p a.txt
8. Вывести строки начиная с первой, где обнаружен шаблон, и до строки x:
Синтаксис: sed -n ‘/pattern/,xp’ filename
Пример: [root@rhel7 ~]# sed -n '/everyone/,5p' a.txt
В данном случае первое вхождение everyone
содержится в строке 3, значит в терминале отобразятся строки с 3 по 5. Если нужно вывести файл до конца, используйте вместо 5 символ $
.
9. Вывести строки с x и до строки, содержащей шаблон. Если шаблон не обнаруживается, выводится файл до конца:
Синтаксис: sed -n ‘x,/pattern/p’ filename
Пример: sed -n '1,/everyone/p' a.txt
10. Вывести все строки, содержащие шаблон, включая следующие за каждой из них x строк:
Синтаксис: sed -n ‘/pattern/,+xp’ filename
Пример: sed -n '/learn/,+2p' a.txt
Замена с помощью команды sed (дополнение)
Ниже приводится небольшой список операций, дополняющий перечисленные ранее.
1. Два способа замены шаблона, игнорируя регистр:
1.1 Использовать флаг /i
:
Синтаксис: sed ‘s/old_pattern/new_pattern/i’ filename
Пример: [root@rhel7 ~]# sed 's/life/Love/i' a.txt
1.2 Использовать регулярные выражения:
Пример: [root@rhel7 ~]# sed 's/[Ll]ife/Love/g' a.txt
2. Замена нескольких пробелов одним:
Пример: [root@rhel7 clang]# sed ‘s/ */ /g’ filename
3. Замена шаблона, сопровождающего другой шаблон:
Синтаксис: sed ‘/followed_pattern/ s/old_pattern/new_pattern/’ filename
Пример: [root@rhel7 ~]# sed '/is/ s/live/love/' a.txt
10. Замена шаблона другим шаблоном, за исключением строки n:
Синтаксис: sed ‘n!s/old_pattern/new_pattern/’ filename
Пример: [root@rhel7 ~]# sed -i '5!s/life/love/' a.txt
Зачем?
Вступление.
Уж я и не знаю, зачем вы это всё читаете, зато прекрасно знаю, для кого это написано: для тех, кому понадобилось научится применять одну из многих UNIX-Way утилит — GNUsed.
Вообще говоря, sed появилась более 30 лет назад, потому споры о том, что это чисто «линуксоидная» фишка, бессмысленны: существуют версии sed для MS-DOS, Windows, MacOS, да и наверное для всех систем где понятие текст имеет хоть какой-то смысл.
Что такое sed? Это (как вам уже наверняка известно) потоковый текстовый редактор. Текст поступает на обработку не в виде файла как обычно, а в виде потока. По этой причине, sed даже более универсальна, чем все остальные редакторы. Утилита sed может работать без участия оператора, без возможности вывода текста, на системах где вывод файлов вообще не возможен, и даже там, где не существует самих файлов! Достаточно всего-лишь потоков, а уж они есть практически везде.
Основные понятия.
Программы должны уметь обмениваться информацией. Это аксиома. Если программы этого не умеют, это не программы, а набор разбухших монстров, которые еле ворочаются на самых современных компьютерах. Примеры всем известны. Проблема только в том, что для обмена нужно выработать правила взаимодействия, в основном, в каком виде обмениваться? Много лет назад, ещё во времена UNIX, было решено обмениваться исключительно текстовой информацией, и появилось соглашение, что считать текстом, а что — нет. Конечно, многие программы используют какие-то особые, специальные форматы, однако, обмениваться ими в общем случае невозможно (если конечно нет подходящего конвертера спец-формат<->текст). Соглашения следующие:
- Текст — это упорядоченный набор строк. Строк может быть любое количество, в том числе их может не быть вовсе.
- Строка — это упорядоченный набор символов. Каждая строка заканчивается особым символом ‘\n’.
- Символ — это минимальный неделимый элемент текста.
- Алфавитом называется список всех допустимых символов в тексте.
Особый случай — это «чистый текст(plain-text)», алфавит в нём состоит из байтов 32…127, 10, и 9.
Если пользователь, а тем более программист не согласен с этими соглашениями, то у него возникают проблемы.
Замечание
Примеров — масса: я сам, помнится, расстраивался, что у меня «не работает» планировщик — я дописал к его таблице свою строчку, однако планировщик меня тупо игнорировал. Всё просто, не было никакой строчки, была фигня непонятная! Я нарушил пункт 2 — не добавил в конец строки ‘\n’. Хотя это скорее беда моего редактора, просил-ведь: «сохрани как текст«, а он…
Замечание
По поводу последнего предложения: Как и все юзеры, я готов обвинять что угодно, кроме собственной криворукости
Особо отмечу проблемы с текстами в UTF-8. В них символы могут быть в несколько байтов, но это ещё не самое страшное (нас это обычно не сильно волнует, это проблема разработчиков sed, мы можем заметить разве-что сильное уменьшение быстродействия). Самое противное — в «тексте» могут появляться несимволы, что приводит к совершенно неожиданным результатам.
История(программа ed).
Первую свою программу я написал лет 20 назад. Однако историю я знаю ещё глубже — дело в том, что в то время (впрочем как и сейчас) новейший супер-компьютер мне никто не дал, пришлось мучиться на довольно старой технике. Именно по этому я и знаю, какими были тогда текстовые редакторы.
Замечание
Кстати, они и сейчас есть: если у вас любой Linux просто наберите ed. Сразу скажу: что-бы выйти из этого чуда, надо набрать команду q, что впрочем относится ко многим утилитам, в т.ч. и к sed.
Основная (для меня) прелесть Linux заключается в том, что мы можем всё сразу попробовать. Достаточно открыть второе окно (вторую консоль для некромантов), и там набирать команды. ИМХО, если у вас хорошо развита моторная память, вам лучше команды именно набирать, тем-же, у кого более развита зрительная — можно копировать мышкой (кстати, я именно так и делаю).
Внимание
Деструктивных примеров здесь не будет конечно, однако, будьте осторожны: не нужно экспериментировать с правами root’а в каталоге со сверх-ценными данными. Все тестовые файлы могут быть изменены, что намного хуже, чем если-бы они были просто удалены. Посему, создайте каталог для тестов, и уж там развлекайтесь. Я вас предупредил.
Раз уж зашла речь о программах, давайте напишем программу на C.
Пример 1.1. Программа test.c.
#include <stdio.h> /* comment */ int main() { printf("тестовый файл\n"); return 0; }
Что делает эта программа не столь важно (если вам интересно — то всё что она сделает будучи скомпилированной — выведет на экран фразу «тестовый файл» и завершит работу), нам интересна эта программа всего-лишь как образец текста. Попробуем отредактировать эту программу так, как это делал я более 20 лет назад:
$ed test.c 90
программа ed написала 90 и что-то ждёт. Ну 90 — это размер файла для редактирования, а ждёт наш редактор команды… Первое что хочется сделать — просмотреть файл. Для этого существует команда p, которая выведет нужные нам строки. Какие именно? Это нужно указать явно, по умолчанию будет выведена только последняя строка, записав 1,$p мы выведем все строки нашего файла. Тут 1,$ означает диапазон строк, от первой строки, и до последней (последняя строка обозначается $
). Это выражение называется адресным, либо просто — адресом, так-как оно определяет, к каким именно строкам будет применена данная команда.
Замечание
На самом деле, адресные выражения намного более разнообразны (и в sed, и в ed), но для ed я не смог найти выражения, что-бы вывести весь текст сразу. Только через несколько минут до меня дошло, что это просто не нужно — реальные тексты состоят из тысяч строк, потому и нет таких адресов — что-бы всё и сразу. Ну выведется 10000 строк, мы их всё равно не успеем прочитать (парадокс: древние компьютеры работали намного быстрее).
Теперь отредактируем наш файл, к примеру заменим «return 0;», на «return 77;». Как всегда, для этого существует множество путей, наверное самый простой, это заново набрать всю строчку. Команда замены строчек у нас есть, называется она c; однако, нужно ещё и указать, к чему-же её применить (к какой именно строке). Проще всего указать номер этой строки, но его тоже надо узнать (хотя можно и посчитать, но это не наш метод), наберём 1,$n:
1 #include <stdio.h> 2 3 /* comment */ 4 5 int main() 6 { 7 printf("тестовый файл\n"); 8 return 0; 9 } 10
Эта команда сработала так-же как p, только перед каждой строкой вывелся её номер, и теперь ясно, что нам нужно изменить восьмую строчку. Для этого наберём 8c, и теперь наберём строчку по новой, нажмём ENTER и CTRL+C. Просмотрев файл, мы убедимся, что замена выполнена.
Согласитесь, неудобно. Однако, в те далёкие семидесятые годы прошлого века ничего другого придумать было нельзя, текст на экране терминала безвозвратно уползал вверх, и мы видели только последние 24 строки (в 25й вводилась команда). Навигация по тексту как в современных редакторах была невозможна, потому пришлось придумать другие методы. Впрочем тут и придумывать ничего не надо: просто нужно заменить в строке 8 число 0 на число 77. Для этого (кто-бы мог подумать) имеется команда замены. Я долго не понимал, почему эта команда называется s, а не более логично, например replace. Точно это мне неизвестно до сих пор, но думается, что главное в этой команде вовсе не замена, а поиск (search), нам сначала надо найти число 0, и уж затем, если найдём конечно, его заменить. Записывается это просто: 8s/0/77/, т.е. «в восьмой строке нужно заменить 0 на 77».
Замечание
Мне подсказали, почему команда s называется именно так: от слова substitution.
Но на практике и это неудобно — проще было-бы не нам смотреть номер строки с «return», а заставить это сделать компьютер. И конечно это ещё тогда было реализовано: в качестве адреса можно задать не только номер, но и саму строку, или даже часть строки, наберём к примеру /ret/n:
8 return 0;
Как видите, ed самостоятельно находит строку с /ret/
, и выполняет для неё заданную команду.
Замечание
Можно задавать диапазоны вроде /begin/,/end/
, для всех строк начиная со строки со словом «begin», и кончая строкой со словом «end», либо к примеру /start/,$
, что значит: от строки со словом «start» и до конца текста. Например (для нашей программы):
/{/,/}/p
{ printf("тестовый файл\n"); return 77; }
Если диапазон(ы) не найден(ы), то ed среагирует вопросиком, как на неправильную команду (забегая вперёд, скажу что sed действует по другому).
Где применяется sed.
Редактор ed конечно не самая удобная штука, но на оборудовании 70х ничего лучше придумать было нельзя. К счастью, прогресс не стоит на месте, и с развитием железа развивались и программы для него, в частности текстовые редакторы. Все вы прекрасно знаете, во что превратились сегодня текстовые редакторы — достаточно сказать, что даже самые опытные пользователи попросту не знают 80% возможностей своего любимого редактора! Однако, была и другая ветка развития текстовых редакторов: во многих случаях нет нужды каждый раз писать новые файлы с нуля, часто нужно просто исправлять какой-либо файл по известному алгоритму, это действие конечно можно произвести обычным редактором, но на самом деле, присутствие юзера тут не только не необходимо, но даже вредно! Юзерам свойственно постоянно ошибаться, всё путать да и вообще — юзер, ленивое и капризное существо: если ему было сказано: «сотри в файле %WINDOWS%\sysok.ini все слова hide», а юзер стёр не все, не hide, да и вообще не то(хорошо, если слова в файлах стирал, а не сами файлы), то этот самый юзер будет обвинять меня, вас, глючный компьютер, кривую программу, карму, да что угодно, но только не свою криворукость…
Имеется ещё 1000 и 7 разных применений, где участие юзера нужно если не исключить, то хотя-бы сильно ограничить: например огромный класс задач про безопасность, ну сколько раз можно повторять: «Читай логи!!!», дык пробовал, «ниасилил — многабукв». Правильно, что-бы юзер увидел опасность необходимо отфильтровать весь «мусор», компьютеру всё равно, он железный: было 10482 атаки на закрытую уязвимость, и одна успешная на открытую — в логе будет 10483 строки (в лучшем случае), и даже если юзер — терпеливая и прилежная блондинка, то даже тогда, она скорее сломает PageDown (сотрёт колесо мыши), прежде чем найдёт нужную строчку.
Ну по поводу фильтров можно возразить: дескать в том-же ed можно набрать g/re/p, что приведёт к фильтрации только нужных строк. Та-же функциональность присутствует во всех остальных текстовых редакторах (включая notepad), мало того, для этого и редактор не нужен, можно прямо в консоли набрать grep (ну вы поняли, почему эта команда так названа). Всё это так, но для фильтрации этого зачастую мало, лично мне по любому не осилить ни 10483 не фильтрованные, ни 2174 фильтрованные строки: до того, как я научился использовать sed, мне приходилось составлять зубодробительные конвейеры, а потом всё равно юзать less|PageDown|search, в разных комбинациях. Это не очень сложно, но отнимает массу времени.
В фильтрации, да и не только в ней, часто нужно реализовать очень простой алгоритм, который почему-то не реализован в различных имеющихся утилитах. Писать для этого программу на C лениво да и не быстро. Для таких случаев придуманы скриптовые языки программирования(ЯП), например bash, perl, awk, php, и многие другие. Все они имеют два недостатка:
- Они универсальные, а значит тянут за собой множество ненужного функционала. Конечно, это удобно для кодера, но вот пользователю это совсем не нужно. Ему надо просто решение, а не «самый крутой ЯП».
-
Все эти ЯП жутко медленные, что связано и с п1, и с тем, что каждую команду нужно множество раз «парсить». Вот например bash’евское
#!/bin/bash echo $VARIABLE
Ага. На 10000 строк! Это-ж каждый раз надо прочитать из файла слово «echo», разобраться что оно значит, а потом найти нужную $VARIABLE… Такой скрипт будет выполнятся в сотни раз дольше чем сам вывод!
Таким образом, нам необходим особый ЯП, который лишён этих недостатков. Таким языком как раз и является sed.
Итог.
Компьютеры предназначены для обработки информации, она может представляются в разных видах, но нас интересует только то, что представлено в виде текста, рассмотрим задачи обработки текста в порядке увеличения сложности:
-
Примитивные задачи.
С такими задачами справятся доступные утилиты, если нам нужно просто скопировать информацию, или скопировать несколько файлов в один, нам нужна cat, для резки файлов подойдёт split и cut, для поиска подстрок сгодится grep, и так далее.
-
Задачи, для которых не хватает инструментов из п1
А вот тут-то и пригодится sed.
-
Ещё более сложные задачи.
Если мощности sed не хватает, придётся писать собственный анализатор текста. При этом не стоит забывать про sed — можно сэкономить огромное количество времени (как своего, так и процессорного), если прогнать входные и/или выходные данные через sed-фильтр(ы).
Подсказка
Что делать, если мощности и/или возможностей возможностей sed не хватает? Следует выяснить почему так. К примеру: ваша задача требует обработки текста, и ещё чего-то, что в sed не реализовано. У вас есть следующие варианты:
- Использовать sed как входной фильтр. К примеру можно не просто посмотреть текст командой less, а посмотреть уже фильтрованный текст.
- Вы можете вызвать другую утилиту внутри sed. Это очень мощный, и потому опасный приём. Далее приведены множество примеров с использованием этой возможности.
- Вы можете выполнить sed внутри своего приложения. Впрочем, мне не кажется, что это — хорошая идея.
- И наконец, вы можете вообще отказаться от sed. Однако, я думаю, очень многие материалы этой книги будут вам полезны, во первых вы сможете быстро и просто протестировать регулярное выражение с помощью sed, а во вторых, речь у меня идёт именно о встроенных в GNU Linux выражениях, а не о каких-то других, которых у вас может и не оказаться под рукой. Вот пример использования регулярных выражений в языке Си (Правда там обсуждается проблема с UTF, но пример как раз и приведён здесь потому, что это проблема касается не только sed, но и всех остальных ЯП).
-
Недоступные для решения на компьютере задачи.
Да, есть и такие… Что-ж, вот тут-то и пригодятся обычные редакторы, например этот текст я набирал в kate. Однако, многие части у меня уже были готовые, потому я их прямо суда вставил, прогнав через sed конечно.
Вы можете обсудить этот документ на форуме. Текст предоставляется по лицензии GNU Free Documentation License (Перевод лицензии GFDL).
Вы можете пожертвовать небольшую сумму яндекс-денег на счёт 41001666004238 для оплаты хостинга, интернета, и прочего. Это конечно добровольно, однако это намного улучшит данный документ (у меня будет больше времени для его улучшения). На самом деле, проект часто находится на грани закрытия, ибо никаких денег никогда не приносил, и приносить не будет. Вы можете мне помочь. Спасибо.
������������� ������� ��������� ��������� ���������� (man-��)
sed (1)
NAME
sed - stream editor
SYNOPSIS
/usr/bin/sed [-n] script [file]...
/usr/bin/sed [-n] [-e script]... [-f script_file]... [file]...
/usr/xpg4/bin/sed [-n] script [file]...
/usr/xpg4/bin/sed [-n] [-e script]... [-f script_file]... [file]...
DESCRIPTION
The sed utility
is a stream editor that reads one or more text files, makes editing changes according to a script of editing commands, and writes the results to standard output. The script is obtained from either the script operand string, or a combination of the option-arguments from the -e script and -f script_file options.
The sed utility is a text editor. It cannot edit binary files or files containing ASCII NUL (\0) characters or very long lines.
OPTIONS
The following options are supported:
-e script
-
script is an edit command for sed. See USAGE below for more information on the format of script. If there is just one -e option and no -f options, the flag -e may be omitted.
-f script_file
-
Takes the script from script_file. script_file consists of editing commands, one per line.
-n
-
Suppresses the default output.
Multiple -e and -f options may be specified. All commands are added to the script in the order specified, regardless of their origin.
OPERANDS
The following operands are supported:
file
-
A path name of a file whose contents will be read and edited. If multiple file operands are specified, the named files will be read in the order specified and the concatenation will be edited. If no file operands are specified, the standard input will be used.
script
-
A string to be used as the script of editing commands. The application must not present a script that violates the restrictions of a text file except that the final character need not be a NEWLINE character.
USAGE
A script consists of editing commands, one per line, of the following form:
[ address [ , address ] ] command [ arguments ]
Zero or more blank characters are accepted before the first address and before command. Any number of semicolons are accepted before the first address.
In normal operation, sed cyclically copies a line of input (less its terminating NEWLINE character) into a pattern space (unless there is something left after a D command), applies in sequence all commands whose addresses select that pattern space, and copies the resulting pattern space to the standard output (except under -n) and deletes the pattern space. Whenever
the pattern space is written to standard output or a named file, sed will immediately follow it with a NEWLINE character.
Some of the commands use a hold space to save all or part of the pattern space for subsequent retrieval. The pattern and hold spaces will each be able to hold at least 8192 bytes.
sed Addresses
An address is either empty, a decimal number that counts input lines cumulatively across files, a $ that addresses the last line of input, or a context address, which consists of a /regular expression/ as
described on the regexp(5) manual page.
A command line with no addresses selects every pattern space.
A command line with one address selects each pattern space that matches the address.
A command line with two addresses selects the inclusive range from the first pattern space that matches the first address through the next pattern space that matches the second address. Thereafter the process is repeated, looking again for the first address. (If the second address is a number less than or equal to the line number selected by the first address, only the line corresponding to the first address is selected.)
Typically, address are separated from each other by a comma (,). They may also be separated by a semicolon (;).
sed Regular Expressions
sed supports the basic regular expressions described on the regexp(5) manual page, with the following additions:
\cREc
-
In a context address, the construction \cREc, where c is any character other than a backslash or NEWLINE character, is identical to /RE/. If the character designated by c appears following a backslash, then it is considered to be that literal character, which does not terminate
the RE. For example, in the context address \xabc\xdefx, the second x stands for itself, so that the regular expression is abcxdef.
\n
-
The escape sequence \n matches a NEWLINE character embedded in the pattern space. A literal NEWLINE character must not be used in the regular expression of a context address or in the substitute command.
Editing commands can be applied only to non-selected pattern spaces by use of the negation command ! (described below).
sed Editing Commands
In the following list of functions the maximum number of permissible addresses for each function is indicated.
The r and w commands take an optional rfile (or wfile) parameter, separated from the command letter by one or more blank characters.
Multiple commands can be specified by separating them with a semicolon (;) on the same command line.
The text argument consists of one or more lines, all but the last of which end with \ to hide the NEWLINE. Each embedded NEWLINE character in the text must be preceded by a backslash. Other backslashes in text are removed and the following character is treated literally. Backslashes in text are treated like backslashes in the replacement string of an s command, and may be used to protect
initial blanks and tabs against the stripping that is done on every script line. The rfile or wfile argument must terminate the command line and must be preceded by exactly one blank. The use of the wfile parameter causes that file to be initially created, if it does not exist, or will replace the contents of an existing file. There can be at most 10 distinct wfile arguments.
Regular expressions match entire strings, not just individual lines, but a NEWLINE character is matched by \n in a sed RE. A NEWLINE character is not allowed in an RE. Also notice that \n cannot be used to match a NEWLINE character at the end of an input line; NEWLINE characters appear in the pattern space as a result of the N editing
command.
Two of the commands take a command-list, which is a list of sed commands separated by NEWLINE characters, as follows:
{ command command }
The { can be preceded with blank characters and can be followed with white space. The commands can be preceded by white space. The terminating } must be preceded by a NEWLINE character and can be preceded or followed by <blank>s. The braces may be preceded or followed by <blank>s. The command may be preceded by <blank>s, but may not be followed by <blank>s.
The following table lists the functions, with the maximum number of permissible addresses.
Max Address | Command | Description |
|
||
1 | a\ text |
Append by executing N command or beginning a new cycle. Place text on the output before reading the next input line. |
|
||
2 | ||
|
||
2 | ||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
1 | ||
|
||
2 | ||
|
||
|
||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
1 | ||
|
||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
2 | ||
|
||
0 | ||
|
||
1 | ||
|
||
2 | ||
0 | ||
|
||
0 |
Max Addr | Command (Using strings) and Description |
|
|
2 | s/regular expression/replacement/flags |
|
|
|
|
2 | |
string2 / | |
See largefile(5) for the description of the behavior of sed when encountering files greater than or equal to 2 Gbyte ( 2^31 bytes).
EXAMPLES
Example 1 An example sed script
This sed script simulates the BSD cat -s command, squeezing excess blank lines from standard input.
sed -n ' # Write non-empty lines. /./ { p d } # Write a single empty line, then look for more empty lines. /^$/ p # Get next line, discard the held <newline> (empty line), # and look for more empty lines. :Empty /^$/ { N s/.// b Empty } # Write the non-empty line before going back to search # for the first in a set of empty lines. p '
ENVIRONMENT VARIABLES
See environ(5) for descriptions of the following environment variables that affect the execution of sed: LANG, LC_ALL, LC_COLLATE, LC_CTYPE, LC_MESSAGES, and NLSPATH.
EXIT STATUS
The following exit values are returned:
0
-
Successful completion.
>0
-
An error occurred.
ATTRIBUTES
See attributes(5) for descriptions of the following attributes:
/usr/bin/sed
ATTRIBUTE TYPE | ATTRIBUTE VALUE |
|
|
Availability | SUNWcsu |
|
|
CSI |
/usr/xpg4/bin/sed
ATTRIBUTE TYPE | ATTRIBUTE VALUE |
|
|
Availability | SUNWxcu4 |
|
|
CSI | |
|
|
Interface Stability |
SEE ALSO
awk(1), ed(1), grep(1), attributes(5), environ(5), largefile(5), regexp(5), standards(5)
Index
- NAME
- SYNOPSIS
- DESCRIPTION
- OPTIONS
- OPERANDS
- USAGE
-
- sed Addresses
- sed Regular Expressions
- sed Editing Commands
- EXAMPLES
- ENVIRONMENT VARIABLES
- EXIT STATUS
- ATTRIBUTES
-
- /usr/bin/sed
- /usr/xpg4/bin/sed
- SEE ALSO
Команда sed — это потоковый редактор текста, работающий по принципу замены. Его можно использовать для поиска, вставки, замены и удаления фрагментов в файле. С помощью этой утилиты вы можете редактировать файлы не открывая их. Будет намного быстрее если вы напишите что и на что надо заменить, чем вы будете открывать редактор vi, искать нужную строку и вручную всё заменять.
В этой статье мы рассмотрим основы использования команды sed linux, её синтаксис, а также синтаксис регулярных выражений, который используется непосредственно для поиска и замены в файлах.
Сначала рассмотрим синтаксис команды:
$ sed опции -e команды файл
А вот её основные опции:
- -n, —quiet — не выводить содержимое буфера шаблона в конце каждой итерации;
- -e — команды, которые надо выполнить для редактирования;
- -f — прочитать команды редактирования из файла;
- -i — сделать резервную копию файла перед редактированием;
- -l — указать свою длину строки;
- -r — включить поддержку расширенного синтаксиса регулярных выражений;
- -s — если передано несколько файлов, рассматривать их как отдельные потоки, а не как один длинный.
Я понимаю, что сейчас всё очень сложно, но к концу статьи всё прояснится.
1. Как работает sed
Теперь нужно понять как работает команда sed. У утилиты есть два буфера, это активный буфер шаблона и дополнительный буфер. Оба изначально пусты. Программа выполняет заданные условия для каждой строки в переданном ей файле.
sed читает одну строку, удаляет из неё все завершающие символы и символы новой строки и помещает её в буфер шаблона. Затем выполняются переданные в параметрах команды, с каждой командой может быть связан адрес, это своего рода условие и команда выполняется только если подходит условие.
Когда всё команды будут выполнены и не указана опция -n, содержимое буфера шаблона выводится в стандартный поток вывода перед этим добавляется обратно символ перевода строки. если он был удален. Затем запускается новая итерация цикла для следующей строки.
Если не используются специальные команды, например, D, то после завершения одной итерации цикла содержимое буфера шаблона удаляется. Однако содержимое предыдущей строки хранится в дополнительном буфере и его можно использовать.
2. Адреса sed
Каждой команде можно передать адрес, который будет указывать на строки, для которых она будет выполнена:
- номер — позволяет указать номер строки, в которой надо выполнять команду;
- первая~шаг — команда будет выполняется для указанной в первой части сроки, а затем для всех с указанным шагом;
- $ — последняя строка в файле;
- /регулярное_выражение/ — любая строка, которая подходит по регулярному выражению. Модификатор l указывает, что регулярное выражение должно быть не чувствительным к регистру;
- номер, номер — начиная от строки из первой части и заканчивая строкой из второй части;
- номер, /регулярное_выражение/ — начиная от сроки из первой части и до сроки, которая будет соответствовать регулярному выражению;
- номер, +количество — начиная от номера строки указанного в первой части и еще плюс количество строк после него;
- номер, ~число — начиная от строки номер и до строки номер которой будет кратный числу.
Если для команды не был задан адрес, то она будет выполнена для всех строк. Если передан один адрес, команда будет выполнена только для строки по этому адресу. Также можно передать диапазон адресов. Тогда адреса разделяются запятой и команда будет выполнена для всех адресов диапазона.
3. Синтаксис регулярных выражений
Вы можете использовать такие же регулярные выражения, как и для Bash и популярных языков программирования. Вот основные операторы, которые поддерживают регулярные выражения sed Linux:
- * — любой символ, любое количество;
- \+ — как звездочка, только один символ или больше;
- \? — нет или один символ;
- \{i\} — любой символ в количестве i;
- \{i,j\} — любой символ в количестве от i до j;
- \{i,\} — любой символ в количестве от i и больше.
4. Команды sed
Если вы хотите пользоваться sed, вам нужно знать команды редактирования. Рассмотрим самые часто применяемые из них:
- # — комментарий, не выполняется;
- q — завершает работу сценария;
- d — удаляет буфер шаблона и запускает следующую итерацию цикла;
- p — вывести содержимое буфера шаблона;
- n — вывести содержимое буфера шаблона и прочитать в него следующую строку;
- s/что_заменять/на_что_заменять/опции — замена символов, поддерживаются регулярные выражения;
- y/символы/символы — позволяет заменить символы из первой части на соответствующие символы из второй части;
- w — записать содержимое буфера шаблона в файл;
- N — добавить перевод строки к буферу шаблона;
- D — если буфер шаблона не содержит новую строку, удалить его содержимое и начать новую итерацию цикла, иначе удалить содержимое буфера до символа перевода строки и начать новую итерацию цикла с тем, что останется;
- g — заменить содержимое буфера шаблона, содержимым дополнительного буфера;
- G — добавить новую строку к содержимому буфера шаблона, затем добавить туда же содержимое дополнительного буфера.
Утилите можно передать несколько команд, для этого их надо разделить точкой с запятой или использовать две опции -e. Теперь вы знаете всё необходимое и можно переходить к примерам.
Примеры использования sed
Теперь рассмотрим примеры sed Linux, чтобы у вас сложилась целостная картина об этой утилите. Давайте сначала выведем из файла строки с пятой по десятую. Для этого воспользуемся командой -p. Мы используем опцию -n чтобы не выводить содержимое буфера шаблона на каждой итерации, а выводим только то, что нам надо. Если команда одна, то опцию -e можно опустить и писать без неё:
sed -n '5,10p' /etc/group
Или можно вывести весь файл, кроме строк с первой по двадцатую:
sed '1,20d' /etc/group
Здесь наоборот, опцию -n не указываем, чтобы выводилось всё, а с помощью команды d очищаем ненужное. Дальше рассмотрим замену в sed. Это самая частая функция, которая применяется вместе с этой утилитой. Заменим вхождения слова root на losst в том же файле и выведем всё в стандартный вывод:
sed 's/root/losst/g' /etc/group
Флаг g заменяет все вхождения, также можно использовать флаг i, чтобы сделать регулярное выражение sed не зависимым от регистра. Для команд можно задавать адреса. Например, давайте выполним замену 0 на 1000, но только в строках с первой по десятую:
sed '1,10 s/0/1000/g' /etc/group
Переходим ещё ближе к регулярным выражениям, удалим все пустые строки или строки с комментариями из конфига Apache:
sed '/^#\|^$\| *#/d' /etc/apache2/apache2.conf
Под это регулярное выражение (адрес) подпадают все строки, которые начинаются с #, пустые, или начинаются с пробела, а за ним идет решетка. Регулярные выражения можно использовать и при замене. Например, заменим все вхождения p в начале строки на losst_p:
sed 's/[$p*]/losst_p/g' /etc/group
Если вам надо записать результат замены в обратно в файл можно использовать стандартный оператор перенаправления вывода > или утилиту tee. Например:
sed '/^#\|^$\| *#/d' /etc/apache2/apache2.conf | sudo tee /etc/apache2/apache2.conf
Также можно использовать опцию -i, тогда утилита не будет выполнять изменения в переданном ей файле:
sudo sed -i '/^#\|^$\| *#/d' /etc/apache2/apache2.conf
Если надо сохранить оригинальный файл, достаточно передать опции -i в параметре расширение для файла резервной копии.
Выводы
Из этой статьи вы узнали что представляет из себя команда sed Linux. Как видите, это очень гибкий инструмент, который позволяет делать с текстом очень многое. Он сложный в освоении, но с помощью него очень удобно решать многие задачи редактирования конфигурационных файлов или фильтрации вывода.
Обнаружили ошибку в тексте? Сообщите мне об этом. Выделите текст с ошибкой и нажмите Ctrl+Enter.
Статья распространяется под лицензией Creative Commons ShareAlike 4.0 при копировании материала ссылка на источник обязательна .