我有这个多行字符串(包括引号):
abc'asdf"
$(dont-execute-this)
foo"bar"''
我将如何分配它到一个变量使用heredoc在Bash?
我需要保留换行符。
我不想转义字符串中的字符,这将是恼人的…
我有这个多行字符串(包括引号):
abc'asdf"
$(dont-execute-this)
foo"bar"''
我将如何分配它到一个变量使用heredoc在Bash?
我需要保留换行符。
我不想转义字符串中的字符,这将是恼人的…
当前回答
从尼尔的回答来看,你通常根本不需要var,你可以像使用变量一样使用函数,而且它比内联或基于读取的解决方案更容易读取。
$ complex_message() {
cat <<'EOF'
abc'asdf"
$(dont-execute-this)
foo"bar"''
EOF
}
$ echo "This is a $(complex_message)"
This is a abc'asdf"
$(dont-execute-this)
foo"bar"''
其他回答
真不敢相信我是第一个发这个的人。
@Erman和@Zombo很接近,但mapfile不只是读取数组…
考虑一下:
#!/bin/bash
mapfile -d '' EXAMPLE << 'EOF'
Hello
こんにちは
今晩は
小夜なら
EOF
echo -n "$EXAMPLE"
收益率:
Hello こんにちは 今晩は 小夜なら
$ "是给mapfile的分隔符,它永远不会出现,它意味着“未分隔”。
因此,不需要无用地使用cat,也不需要重新组合数组。
此外,你还能得到这样的好处:
$ echo $EXAMPLE
Hello こんにちは 今晩は 小夜なら
你没有收到@Zombo的方法:
mapfile y <<'z'
abc'asdf"
$(dont-execute-this)
foo"bar"''
z
echo $y
abc'asdf"
奖金
如果你通过head -c -1运行它,你也可以用一种不会不性能的方式摆脱最后的换行:
unset EXAMPLE
mapfile -d '' EXAMPLE < <(head -c -1 << EOF
Hello
こんにちは
今晩は
小夜なら
EOF
)
printf "%q" "$EXAMPLE"
$'Hello\nこんにちは\n今晩は\n小夜なら'
$TEST="ok"
read MYTEXT <<EOT
this bash trick
should preserve
newlines $TEST
long live perl
EOT
echo -e $MYTEXT
这是Dennis方法的变化,在脚本中看起来更优雅。
函数定义:
define(){ IFS='\n' read -r -d '' ${1} || true; }
用法:
define VAR <<'EOF'
abc'asdf"
$(dont-execute-this)
foo"bar"''
EOF
echo "$VAR"
享受
附注:为不支持read -d的shell创建了一个“read循环”版本。应该适用于set -eu和未配对的反引号,但测试不太好:
define(){ o=; while IFS="\n" read -r a; do o="$o$a"'
'; done; eval "$1=\$o"; }
使用$()将cat的输出赋值给你的变量,如下所示:
VAR=$(cat <<'END_HEREDOC'
abc'asdf"
$(dont-execute-this)
foo"bar"''
END_HEREDOC
)
# this will echo variable with new lines intact
echo "$VAR"
# this will echo variable without new lines (changed to space character)
echo $VAR
确保从END_HEREDOC开始用单引号分隔。
注意,结束括号分隔符END_HEREDOC必须单独在一行上(因此结束括号在下一行上)。
感谢@ephemient的回答。
数组是一个变量,所以在这种情况下mapfile可以工作
mapfile y <<'z'
abc'asdf"
$(dont-execute-this)
foo"bar"''
z
然后你可以像这样打印
printf %s "${y[@]}"