printf — форматирование и вывод данных

В отличие от других команд в этой главе, команда printf не используется в конвейерной обработке (она не принимает данные со стандартного ввода) и редко применяется непосредственно в командной строке (чаще она используется в сценариях). Почему это так важно? Потому что она используется очень широко.

Команда printf (ее название происходит от print formatted (форматированный вывод)) первоначально была создана как функция для языка программирования C и впоследствии была реализована во многих других языках, включая язык командной оболочки. Фактически в bash команда printf реализована как встроенная команда. Она имеет следующий синтаксис:

printf "формат" аргументы

Команда принимает строку с описанием формата, которая затем применяется к списку аргументов. Отформатированный результат передается в стандартный вывод. Ниже приводится простой пример:

[me@linuxbox ~]$ printf "I formatted the string: %s " foo

I formatted the string: foo

Строка формата может содержать литеральный текст (такой, как I formatted the string:), экранированные последовательности (такие, как , символ перевода строки) и последовательности, начинающиеся с символа %, которые называют спецификаторами преобразований (conversion specifications). В примере выше спецификатор преобразования %s используется для форматирования строки foo и включения ее в вывод команды. Еще один пример:

[me@linuxbox ~]$ printf "I formatted '%s' as a string. " foo

I formatted 'foo' as a string.

Как видите, в выводе команды спецификатор преобразования %s замещается строкой foo. Преобразование s используется для форматирования строковых данных. Существуют также другие спецификаторы для других видов данных. В табл. 21.4 перечислены наиболее часто используемые типы данных.

Таблица 21.4. Спецификаторы типов данных, наиболее часто используемых в команде printf

Спецификатор

Описание

d

Форматирует число как десятичное целое со знаком

f

Форматирует и выводит вещественное число

o

Форматирует целочисленное значение как восьмеричное число

s

Форматирует строку

x

Форматирует целочисленное значение как шестнадцатеричное число, с использованием букв a—f нижнего регистра, где это необходимо

X

То же, что и x, но использует буквы верхнего регистра

%

Выводит литеральный символ % (то есть сам спецификатор имеет вид: %%)

Продемонстрируем действие каждого спецификатора преобразования на примере строки 380:

[me@linuxbox ~]$ printf "%d, %f, %o, %s, %x, %X " 380 380 380 380 380 380

380, 380.000000, 574, 380, 17c, 17C

Так как в строке формата указано шесть спецификаторов формата, нам потребовалось передать команде printf шесть аргументов. Шесть результатов показывают результат действия каждого спецификатора.

Для настройки вывода спецификатору формата можно передать несколько дополнительных компонентов. Полный синтаксис спецификатора преобразования выглядит так:

%[флаги][ширина][.точность]спецификатор_преобразования

Для правильной интерпретации дополнительные компоненты, если их несколько, должны передаваться в указанном порядке. Все компоненты описаны в табл. 21.5.

Таблица 21.5. Компоненты спецификаторов преобразований в команде printf

Спецификатор

Описание

флаги

Существует пять разных флагов:

# использовать альтернативный формат вывода — действует по-разному, в зависимости от типа данных. Для преобразования o (восьмеричное число) в вывод добавляется префикс 0 (ноль). Для преобразований x и X (шестнадцатеричное число) в вывод добавляется префикс 0x или 0X соответственно.

0 (ноль) — добавляет нули в начало вывода. Это означает, что поле будет дополнено ведущими нулями, например: 000380.

- (дефис) — выравнивание по левому краю. По умолчанию printf выполняет выравнивание по правому краю.

(пробел) — добавляет ведущий пробел перед положительными числами.

+ (знак «плюс») — выводит знак перед положительными числами. По умолчанию printf выводит знаки только перед отрицательными числами

ширина

Число, определяющее минимальную ширину поля вывода в символах

.точность

Определяет число знаков после десятичной запятой при выводе вещественных чисел. Для строковых значений точность определяет число выводимых символов

В табл. 21.6 перечислены некоторые примеры применения разных форматов.

Таблица 21.6. Примеры применения спецификаторов преобразований команды printf

Аргумент

Формат

Результат

Примечания

380

"%d"

380

Простое форматирование целых чисел

380

"%#x"

0x17c

Форматирование целочисленных значений в шестнадцатеричное представление с использованием альтернативного форматирования

380

"%05d"

00380

Форматирование целочисленных значений с ведущими нулями и минимальным размером поля, равным пяти символам

380

"%05.5f"

380.00000

Форматирование вещественных значений с ведущими нулями и 5 знаками после запятой. Поскольку указанная минимальная ширина поля (5) меньше фактической ширины отформатированного числа, ведущие нули не были добавлены

380

"%010.5f"

0380.00000

Ширина поля вывода увеличена до 10, вследствие чего появился ведущий нуль

380

"%+d"

+380

Флаг + требует выводить знак у положительных чисел

380

"%-d"

380

Флаг - обеспечивает форматирование с выравниванием по левому краю

abcdefghijk

"%5s"

abcdefghijk

Форматирование строки в поле с указанной минимальной шириной

abcdefghijk

"%.5s"

фисву

Применение компонента точности к строке привело к ее усечению

И еще раз: команда printf в основном используется в сценариях, где применяется для форматирования табличных данных, а не как самостоятельный инструмент командной строки. Тем не менее мы можем использовать ее для решения различных задач форматирования. Во-первых, попробуем вывести несколько полей, разделив их символами табуляции:

[me@linuxbox ~]$ printf "%s %s %s " str1 str2 str3

str1 str2 str3

Добавив (экранированную последовательность, соответствующую символу табуляции), мы достигли желаемого эффекта. Затем попробуем вывести несколько чисел в форматированном виде:

[me@linuxbox ~]$ printf "Line: %05d %15.3f Result: %+15d " 1071 3.14156295 32589

Line: 01071 3.142 Result: +32589

Здесь демонстрируется действие компонента, определяющего минимальную ширину поля. А можно ли подобным образом отформатировать небольшую веб-страницу?

[me@linuxbox ~]$ printf "<html> <head> <title>%s</title> </head>

<body> <p>%s</p> </body> </html> " "Page Title" "Page Content"

<html>

<head>

<title>Page Title</title>

</head>

<body>

<p>Page Content</p>

</body>

</html>