Awk — это язык сценариев общего назначения, предназначенный для расширенной обработки текста. В основном он используется как инструмент отчетности и анализа.
В отличие от большинства других процедурных языков программирования, awk управляется данными, что означает, что вы определяете набор действий, выполняемых с вводимым текстом. Он принимает входные данные, преобразует их и отправляет результат на стандартный вывод.
В этой статье рассматриваются основы языка программирования awk. Знание основ awk значительно улучшит вашу способность манипулировать текстовыми файлами в командной строке.
Как работает awk
Существует несколько различных реализаций awk. Мы будем использовать GNU-реализацию awk, которая называется gawk. В большинстве систем Linux интерпретатор awk
— это просто символическая ссылка на gawk
.
Записи и поля
Awk может обрабатывать текстовые файлы данных и потоки. Входные данные разделены на записи и поля. Awk работает с одной записью за раз, пока не будет достигнут конец ввода. Записи разделяются символом, который называется разделителем записей. Разделителем записей по умолчанию является символ новой строки, что означает, что каждая строка в текстовых данных является записью. Новый разделитель записей может быть установлен с помощью переменной RS
.
Записи состоят из полей, разделенных разделителем полей. По умолчанию поля разделяются пробелом, включая один или несколько символов табуляции, пробела и новой строки.
Поля в каждой записи обозначаются знаком доллара ( $
), за которым следует номер поля, начинающийся с 1. Первое поле представлено с помощью $1
, второе — с помощью $2
и так далее. На последнее поле также можно ссылаться с помощью специальной переменной $NF
. На всю запись можно ссылаться с помощью $0
.
Вот визуальное представление, показывающее, как ссылаться на записи и поля:
tmpfs 788M 1.8M 786M 1% /run/lock
/dev/sda1 234G 191G 31G 87% /
|-------| |--| |--| |--| |-| |--------|
$1 $2 $3 $4 $5 $6 ($NF) --> fields
|-----------------------------------------|
$0 --> record
Программа awk
Чтобы обработать текст с помощью awk
, вы пишете программу, которая сообщает команде, что делать. Программа состоит из ряда правил и пользовательских функций. Каждое правило содержит одну пару шаблон и действие. Правила разделяются новой строкой или точкой с запятой ( ;
). Обычно awk-программа выглядит так:
pattern { action }
pattern { action }
...
Когда awk
обрабатывает данные, если шаблон соответствует записи, он выполняет указанное действие с этой записью. Если у правила нет шаблона, все записи (строки) совпадают.
Действие awk заключено в фигурные скобки ( {}
) и состоит из операторов. Каждый оператор определяет операцию, которую нужно выполнить. В действии может быть несколько операторов, разделенных новой строкой или точкой с запятой ( ;
). Если правило не имеет действия, по умолчанию выполняется печать всей записи.
Awk поддерживает различные типы операторов, включая выражения, условные операторы, операторы ввода, вывода и т. Д. Наиболее распространенные операторы awk:
exit
— останавливает выполнение всей программы и выходит.next
— останавливает обработку текущей записи и переходит к следующей записи во входных данных.print
— Печать записей, полей, переменных и настраиваемого текста.printf
— дает вам больше контроля над форматом вывода, аналогично C и bashprintf
.
При написании программ awk все, что находится после решетки (#)
и до конца строки, считается комментарием. Длинные строки можно разбить на несколько строк с помощью символа продолжения, обратной косой черты ( ).
Выполнение программ awk
Программа awk может быть запущена несколькими способами. Если программа короткая и простая, ее можно передать непосредственно интерпретатору awk
из командной строки:
awk 'program' input-file...
При запуске программы в командной строке ее следует заключать в одинарные кавычки ( ''
), чтобы оболочка не интерпретировала программу.
Если программа большая и сложная, лучше всего поместить ее в файл и использовать параметр -f
для передачи файла команде awk
:
awk -f program-file input-file...
В приведенных ниже примерах мы будем использовать файл с именем «team.txt», который выглядит примерно так:
Bucks Milwaukee 60 22 0.732
Raptors Toronto 58 24 0.707
76ers Philadelphia 51 31 0.622
Celtics Boston 49 33 0.598
Pacers Indiana 48 34 0.585
Шаблоны AWK
Шаблоны в awk определяют, следует ли выполнять соответствующее действие.
Awk поддерживает различные типы шаблонов, включая регулярное выражение, выражение отношения, диапазон и шаблоны специальных выражений.
Если у правила нет шаблона, сопоставляется каждая входная запись. Вот пример правила, содержащего только действие:
awk '{ print $3 }' teams.txt
Программа распечатает третье поле каждой записи:
60
58
51
49
48
Шаблоны регулярных выражений
Регулярное выражение или регулярное выражение — это шаблон, который соответствует набору строк. Шаблоны регулярных выражений AWK заключаются в косую черту ( //
):
/regex pattern/ { action }
Самый простой пример — это буквальное сопоставление символа или строки. Например, чтобы отобразить первое поле каждой записи, содержащее «0,5», вы должны выполнить следующую команду:
awk '/0.5/ { print $1 }' teams.txt
Celtics
Pacers
Шаблон может быть любым типом расширенного регулярного выражения. Вот пример, который печатает первое поле, если запись начинается с двух или более цифр:
awk '/^[0-9][0-9]/ { print $1 }' teams.txt
76ers
Шаблоны реляционных выражений
Шаблоны реляционных выражений обычно используются для сопоставления содержимого определенного поля или переменной.
По умолчанию шаблоны регулярных выражений сопоставляются с записями. Чтобы сопоставить регулярное выражение с полем, укажите поле и используйте оператор сравнения «содержать» ( ~
) с шаблоном.
Например, чтобы напечатать первое поле каждой записи, второе поле которой содержит «ia», вы должны ввести:
awk '$2 ~ /ia/ { print $1 }' teams.txt
76ers
Pacers
Чтобы сопоставить поля, которые не содержат заданного шаблона, используйте оператор !~
:
awk '$2 !~ /ia/ { print $1 }' teams.txt
Bucks
Raptors
Celtics
Вы можете сравнивать строки или числа для таких отношений, как, больше, меньше, равно и т. Д. Следующая команда печатает первое поле всех записей, третье поле которых больше 50:
awk '$3 > 50 { print $1 }' teams.txt
Bucks
Raptors
76ers
Шаблоны диапазонов
Шаблоны диапазонов состоят из двух шаблонов, разделенных запятой:
pattern1, pattern2
Все записи, начинающиеся с записи, соответствующей первому шаблону, до совпадения с записью, соответствующей второму шаблону.
Вот пример, который напечатает первое поле всех записей, начиная с записи, включая «Raptors», до записи, включающей «Celtics»:
awk '/Raptors/,/Celtics/ { print $1 }' teams.txt
Raptors
76ers
Celtics
Шаблоны также могут быть выражениями отношений. Приведенная ниже команда распечатает все записи, начиная с той, четвертое поле которой равно 32, до той, четвертое поле которой равно 33:
awk '$4 == 31, $4 == 33 { print $0 }' teams.txt
76ers Philadelphia 51 31 0.622
Celtics Boston 49 33 0.598
Шаблоны диапазона нельзя комбинировать с другими выражениями шаблона.
Специальные шаблоны выражения
Awk включает следующие специальные паттерны:
BEGIN
— используется для выполнения действий перед обработкой записей.END
— используется для выполнения действий после обработки записей.
Шаблон BEGIN
обычно используется для установки переменных, а шаблон END
для обработки данных из записей, таких как вычисления.
В следующем примере печатается «Начать обработку.», Затем печатается третье поле каждой записи и, наконец, «Завершить обработку».
awk 'BEGIN { print "Start Processing." }; { print $3 }; END { print "End Processing." }' teams.txt
Start Processing
60
58
51
49
48
End Processing.
Если программа имеет только шаблон BEGIN
, действия выполняются, а ввод не обрабатывается. Если в программе есть только шаблон END
, ввод обрабатывается перед выполнением действий правила.
Версия awk для Gnu также включает еще два специальных шаблона BEGINFILE
и ENDFILE
, которые позволяют выполнять действия при обработке файлов.
Комбинирование узоров
Awk позволяет комбинировать два или более шаблонов, используя логический оператор И ( &&
) и логический оператор ИЛИ ( ||
).
Вот пример, в котором оператор &&
используется для печати первого поля той записи, у которой третье поле больше 50, а четвертое поле меньше 30:
awk '$3 > 50 && $4 < 30 { print $1 }' teams.txt
Bucks
Raptors
Встроенные переменные
Awk имеет ряд встроенных переменных, которые содержат полезную информацию и позволяют управлять обработкой программы. Ниже приведены некоторые из наиболее распространенных встроенных переменных:
NF
— количество полей в записи.NR
— номер текущей записи.FILENAME
— имя входного файла, который в данный момент обрабатывается.FS
— Разделитель полей.RS
— Разделитель записей.OFS
— Разделитель выходных полей.ORS
— разделитель выходной записи.
Вот пример, показывающий, как напечатать имя файла и количество строк (записей):
awk 'END { print "File", FILENAME, "contains", NR, "lines." }' teams.txt
File teams.txt contains 5 lines.
Переменные в AWK могут быть установлены в любой строке программы. Чтобы определить переменную для всей программы, поместите ее в шаблон BEGIN
.
Изменение поля и разделителя записей
По умолчанию значение разделителя полей — любое количество пробелов или символов табуляции. Его можно изменить, установив в переменной FS
.
Например, чтобы установить разделитель полей .
вы бы использовали:
awk 'BEGIN { FS = "." } { print $1 }' teams.txt
Bucks Milwaukee 60 22 0
Raptors Toronto 58 24 0
76ers Philadelphia 51 31 0
Celtics Boston 49 33 0
Pacers Indiana 48 34 0
Разделитель полей также может содержать более одного символа:
awk 'BEGIN { FS = ".." } { print $1 }' teams.txt
При запуске однострочных команд awk в командной строке вы также можете использовать параметр -F
для изменения разделителя полей:
awk -F "." '{ print $1 }' teams.txt
По умолчанию разделителем записей является символ новой строки, который можно изменить с помощью переменной RS
.
Вот пример, показывающий, как изменить разделитель записей на .
:
awk 'BEGIN { RS = "." } { print $1 }' teams.txt
Bucks Milwaukee 60 22 0
732
Raptors Toronto 58 24 0
707
76ers Philadelphia 51 31 0
622
Celtics Boston 49 33 0
598
Pacers Indiana 48 34 0
585
Действия при отсутствии нагрузки
Действия awk заключаются в фигурные скобки ( {}
) и выполняются при совпадении с шаблоном. Действие может иметь ноль или более утверждений. Несколько операторов выполняются в том порядке, в котором они появляются, и должны быть разделены новой строкой или точкой с запятой ( ;
).
В awk поддерживается несколько типов операторов действий:
- Выражения, такие как присваивание переменных, арифметические операторы, операторы увеличения и уменьшения.
- Управляющие операторы, используемые для управления потоком программы (
if
,for
,while
,switch
и т. Д.) - Операторы вывода, такие как
print
иprintf
. - Составные утверждения, чтобы сгруппировать другие утверждения.
- Операторы ввода, чтобы управлять обработкой ввода.
- Операторы удаления для удаления элементов массива.
Оператор print
вероятно, является наиболее часто используемым оператором awk. Он печатает форматированный вывод текста, записей, полей и переменных.
При печати нескольких элементов их нужно разделять запятыми. Вот пример:
awk '{ print $1, $3, $5 }' teams.txt
Печатные материалы разделяются одиночными пробелами:
Bucks 60 0.732
Raptors 58 0.707
76ers 51 0.622
Celtics 49 0.598
Pacers 48 0.585
Если вы не используете запятые, между элементами не будет пробелов:
awk '{ print $1 $3 $5 }' teams.txt
Печатные элементы объединены:
Bucks600.732
Raptors580.707
76ers510.622
Celtics490.598
Pacers480.585
Когда print
используется без аргументов, по умолчанию используется print $0
. Текущая запись будет напечатана.
Чтобы напечатать собственный текст, вы должны заключить текст в двойные кавычки:
awk '{ print "The first field:", $1}' teams.txt
The first field: Bucks
The first field: Raptors
The first field: 76ers
The first field: Celtics
The first field: Pacers
Вы также можете печатать специальные символы, такие как новая строка:
awk 'BEGIN { print "First linenSecond linenThird line" }'
First line
Second line
Third line
Оператор printf
дает вам больше контроля над форматом вывода. Вот пример вставки номеров строк:
awk '{ printf "%3d. %sn", NR, $0 }' teams.txt
printf
не создает новую строку после каждой записи, поэтому мы используем n
:
1. Bucks Milwaukee 60 22 0.732
2. Raptors Toronto 58 24 0.707
3. 76ers Philadelphia 51 31 0.622
4. Celtics Boston 49 33 0.598
5. Pacers Indiana 48 34 0.585
Следующая команда вычисляет сумму значений, хранящихся в третьем поле в каждой строке:
awk '{ sum += $3 } END { printf "%dn", sum }' teams.txt
266
Вот еще один пример, показывающий, как использовать выражения и управляющие операторы для печати квадратов чисел от 1 до 5:
awk 'BEGIN { i = 1; while (i < 6) { print "Square of", i, "is", i*i; ++i } }'
Square of 1 is 1
Square of 2 is 4
Square of 3 is 9
Square of 4 is 16
Square of 5 is 25
Однострочные команды, подобные приведенной выше, труднее понять и поддерживать. При написании более длинных программ следует создать отдельный программный файл:
BEGIN {
i = 1
while (i < 6) {
print "Square of", i, "is", i*i;
++i
}
}
Запустите программу, передав имя файла интерпретатору awk
:
awk -f prg.awk
Вы также можете запустить программу awk как исполняемый файл, используя директиву shebang и установив интерпретатор awk
:
#!/usr/bin/awk -f
BEGIN {
i = 1
while (i < 6) {
print "Square of", i, "is", i*i;
++i
}
}
Сохраните файл и сделайте его исполняемым :
chmod +x prg.awk
Теперь вы можете запустить программу, введя:
./prg.awk
Использование переменных оболочки в программах AWK
Если вы используете команду awk
в сценариях оболочки, велика вероятность, что вам потребуется передать переменную оболочки программе awk. Один из вариантов — заключить программу в двойные вместо одинарных кавычек и подставить переменную в программе. Однако эта опция сделает вашу awk-программу более сложной, так как вам нужно будет избежать переменных awk.
Рекомендуемый способ использования переменных оболочки в программах awk — присвоить переменную оболочки переменной awk. Вот пример:
num=51
awk -v n="$num" 'BEGIN {print n}'
51
Выводы
Awk — один из самых мощных инструментов для работы с текстом.
Добавить комментарий