Как я могу создать новую строку в строке в sh?

Это

STR="Hello\nWorld"
echo $STR

производит как выход

Hello\nWorld

вместо

Hello
World

Что мне делать, чтобы иметь строку новой строки в строке?

Примечание: Этот вопрос не касается эха. Я знаю echo -e, но я ищу решение, которое позволяет передавать строку (которая включает в себя новую строку) в качестве аргумента другим командам, которые не имеют аналогичной опции для интерпретации \n как новых строк.

Ответ 1

Решение состоит в использовании $'string', например:

$ STR=$'Hello\nWorld'
$ echo "$STR"
Hello
World

Вот выдержка из справочной страницы Bash:

   Words of the form $'string' are treated specially.  The word expands to
   string, with backslash-escaped characters replaced as specified by  the
   ANSI  C  standard.  Backslash escape sequences, if present, are decoded
   as follows:
          \a     alert (bell)
          \b     backspace
          \e
          \E     an escape character
          \f     form feed
          \n     new line
          \r     carriage return
          \t     horizontal tab
          \v     vertical tab
          \\     backslash
          \'     single quote
          \"     double quote
          \nnn   the eight-bit character whose value is  the  octal  value
                 nnn (one to three digits)
          \xHH   the  eight-bit  character  whose value is the hexadecimal
                 value HH (one or two hex digits)
          \cx    a control-x character

   The expanded result is single-quoted, as if the  dollar  sign  had  not
   been present.

   A double-quoted string preceded by a dollar sign ($"string") will cause
   the string to be translated according to the current  locale.   If  the
   current  locale  is  C  or  POSIX,  the dollar sign is ignored.  If the
   string is translated and replaced, the replacement is double-quoted.

Ответ 2

Эхо настолько девяностые и настолько чревато опасностями, что его использование должно приводить к сбросу ядра не менее 4 ГБ. Серьезно, проблемы эха были причиной того, что процесс стандартизации Unix наконец придумал утилиту printf, устраняя все проблемы.

Итак, чтобы получить строку строки в строке:

FOO="hello
world"
BAR=$(printf "hello\nworld\n") # Alternative; note: final newline is deleted
printf '<%s>\n' "$FOO"
printf '<%s>\n' "$BAR"

Там! Нет SYSV против BSD echo madness, все становится аккуратно напечатанной и полностью переносимой поддержкой для escape-последовательностей C. Теперь всем нравится использовать printf и никогда не оглядываться назад.

Ответ 3

То, что я сделал, основываясь на других ответах, было

NEWLINE=$'\n'
my_var="__between eggs and bacon__"
echo "spam${NEWLINE}eggs${my_var}bacon${NEWLINE}knight"

# which outputs:
spam
eggs__between eggs and bacon__bacon
knight

Ответ 4

Проблема не в оболочке. Проблема заключается в самом деле с командой echo и отсутствием двойных кавычек вокруг интерполяции переменных. Вы можете попробовать использовать echo -e, но это не поддерживается на всех платформах, и теперь одна из причин printf теперь рекомендуется для переносимости.

Вы также можете попробовать вставить новую строку непосредственно в свою оболочку script (если script - это то, что вы пишете), так что это выглядит...

#!/bin/sh
echo "Hello
World"
#EOF

или эквивалентно

#!/bin/sh
string="Hello
World"
echo "$string"  # note double quotes!

Ответ 5

  • Единственная простая альтернатива - фактически набрать новую строку в переменной:

    $ STR='new
    line'
    $ printf '%s' "$STR"
    new
    line
    

    Да, это означает запись Enter в случае необходимости в коде.

  • Существует несколько эквивалентов символа new line.

    \n           ### A common way to represent a new line character.
    \012         ### Octal value of a new line character.
    \x0A         ### Hexadecimal value of a new line character.
    

    Но все они требуют "интерпретации" каким-то инструментом (POSIX printf):

    echo -e "new\nline"           ### on POSIX echo, `-e` is not required.
    printf 'new\nline'            ### Understood by POSIX printf.
    printf 'new\012line'          ### Valid in POSIX printf.
    printf 'new\x0Aline'       
    printf '%b' 'new\0012line'    ### Valid in POSIX printf.
    

    И поэтому инструмент должен построить строку с новой строкой:

    $ STR="$(printf 'new\nline')"
    $ printf '%s' "$STR"
    new
    line
    
  • В некоторых оболочках последовательность $ ' является специальным расширением оболочки. Известен для работы в ksh93, bash и zsh:

    $ STR=$'new\nline'
    
  • Конечно, возможны и более сложные решения:

    $ echo '6e65770a6c696e650a' | xxd -p -r
    new
    line
    

    Или

    $ echo "new line" | sed 's/ \+/\n/g'
    new
    line
    

Ответ 6

Я считаю флаг -e элегантным и прямым

bash$ STR="Hello\nWorld"

bash$ echo -e $STR
Hello
World

Если строка является выводом другой команды, я просто использую кавычки

indexes_diff=$(git diff index.yaml)
echo "$indexes_diff"

Ответ 7

A $прямо перед одинарными кавычками "...\n..." следующим образом, однако двойные кавычки не работают.

$ echo $'Hello\nWorld'
Hello
World
$ echo $"Hello\nWorld"
Hello\nWorld

Ответ 8

Я не эксперт bash, но этот работал у меня:

STR1="Hello"
STR2="World"
NEWSTR=$(cat << EOF
$STR1

$STR2
EOF
)
echo "$NEWSTR"

Мне было проще форматировать тексты.

Ответ 9

В моей системе (Ubuntu 17.10) ваш пример работает по желанию, как при вводе из командной строки (в sh), так и при выполнении в качестве sh script:

[bash]§ sh
$ STR="Hello\nWorld"
$ echo $STR
Hello
World
$ exit
[bash]§ echo "STR=\"Hello\nWorld\"
> echo \$STR" > test-str.sh
[bash]§ cat test-str.sh 
STR="Hello\nWorld"
echo $STR
[bash]§ sh test-str.sh 
Hello
World

Я думаю, это отвечает на ваш вопрос: он просто работает. (Я не пытался разобраться в деталях, например, в какой момент точно замена символа новой строки для \n происходит в sh).

Тем не менее, я заметил, что этот же script будет вести себя по-другому, когда выполняется с bash, и вместо этого выведет Hello\nWorld:

[bash]§ bash test-str.sh
Hello\nWorld

Мне удалось получить желаемый результат с помощью bash следующим образом:

[bash]§ STR="Hello
> World"
[bash]§ echo "$STR"

Обратите внимание на двойные кавычки вокруг $STR. Это ведет себя одинаково, если сохраняется и выполняется как bash script.

Следующее также дает желаемый результат:

[bash]§ echo "Hello
> World"

Ответ 10

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

IFS="$(printf '\nx')"
IFS="${IFS%x}"

Bash (и, вероятно, другие оболочки) собирают все завершающие символы новой строки после подстановки команд, поэтому вам нужно закончить строку printf символом новой строки и впоследствии удалить его. Это также может легко стать oneliner.

IFS="$(printf '\nx')" IFS="${IFS%x}"

Я знаю, что это два действия, а не одно, но мой отпечаток и переносимость OCD в настоящее время в мире :) Я изначально разработал это, чтобы иметь возможность разделять только разделенные выходные данные, и я закончил использование модификации, которая использует \r как завершающий персонаж. Это делает расщепление новой строки даже для вывода dos, заканчивающегося \r\n.

IFS="$(printf '\n\r')"

Ответ 11

Я не был действительно счастлив ни с одним из вариантов здесь. Это то, что сработало для меня.

str=$(printf "%s" "first line")
str=$(printf "$str\n%s" "another line")
str=$(printf "$str\n%s" "and another line")