如何在Bash中使用变量编写多行字符串?

320

如何使用BASH在名为myconfig.conf的文件中编写多行内容?

#!/bin/bash
kernel="2.6.39";
distro="xyz";

echo <<< EOL
line 1, ${kernel}
line 2,
line 3, ${distro}
line 4
line ...
EOL >> /etc/myconfig.conf;
cat /etc/myconfig.conf;
9个回答

629

语法(<<<)和所使用的命令(echo)是错误的。

正确的方式应该是:

#!/bin/bash

kernel="2.6.39"
distro="xyz"
cat >/etc/myconfig.conf <<EOL
line 1, ${kernel}
line 2, 
line 3, ${distro}
line 4 line
... 
EOL

cat /etc/myconfig.conf

这个结构被称为Here Document(文档),可以在Bash手册的man --pager='less -p "\s*Here Documents"' bash页面找到。


61
翻译从英语到中文。仅返回翻译后的文字内容:如果您要追加,可以使用cat >> - Or Gal
30
这个方法很好用,但你必须确保在结束的 EOF 前面没有空格,否则它将无法被识别,你会遇到一个 意外的文件结尾 错误。 - nwinkler
14
这被称为heredoc。 - William Pursell
5
如果我需要sudo权限才能写入文件怎么办? - gfpacheco
21
你可以使用tee命令实现此功能,例如将文本通过管道输入给tee命令,并使用sudo权限将其写入到/etc/myconfig.conf文件中:cat << EOL | sudo tee /etc/myconfig.conf - Xin Chen
显示剩余8条评论

101
#!/bin/bash
kernel="2.6.39";
distro="xyz";

cat > /etc/myconfig.conf << EOL
line 1, ${kernel}
line 2,
line 3, ${distro}
line 4
line ...
EOL

这将实现你想要的功能。


9
@ktf,我打字速度不比你快,但是输入的字母比你少。^_* - Kent
你速度很快,但还不够 :)))) - Milad Yarmohammadi
这非常有帮助! - Vibhanshu Biswas

53

如果您不希望替换变量,您需要在EOL周围加上单引号。

cat >/tmp/myconfig.conf <<'EOL'
line 1, ${kernel}
line 2, 
line 3, ${distro}
line 4 line
... 
EOL

之前的例子:

$ cat /tmp/myconfig.conf 
line 1, ${kernel}
line 2, 
line 3, ${distro}
line 4 line
... 

非常感谢你在我不想修改变量时给予细微差别的处理。例如,将其放在引号中'EOL' - undefined

19

heredoc方案无疑是最常用的方法。其他常见的解决方案包括:

echo 'line 1, '"${kernel}"'
line 2,
line 3, '"${distro}"'
line 4' > /etc/myconfig.conf

以及

exec 3>&1 # Save current stdout
exec > /etc/myconfig.conf
echo line 1, ${kernel}
echo line 2, 
echo line 3, ${distro}
...
exec 1>&3  # Restore stdout

printf "%s\n" "line1, ${kernel}" "line2," "line3, $distro" ...

也许还可以指向“printf”,这会介绍一些更有趣的变化。 - tripleee

10

我使用的是Mac OS,对于在SH脚本中编写多行代码,以下代码对我有效:

#! /bin/bash
FILE_NAME="SomeRandomFile"

touch $FILE_NAME

echo """I wrote all
the  
stuff
here.
And to access a variable we can use
$FILE_NAME  

""" >> $FILE_NAME

cat $FILE_NAME

请确保按照需要为脚本文件分配chmod权限。 我已经使用过。
chmod u+x myScriptFile.sh

也适用于Ubuntu 20。 - Jonathan

4
以下机制有助于将多行重定向到文件。将完整的字符串放在 " 下面,这样我们就可以重定向变量的值。
#!/bin/bash
kernel="2.6.39"
echo "line 1, ${kernel}
line 2," > a.txt
echo 'line 2, ${kernel}
line 2,' > b.txt

a.txt 的内容是:

line 1, 2.6.39
line 2,

b.txt的内容是:

line 2, ${kernel}
line 2,

0
在我的使用情况中,我不得不使用以下内容:
{
 ...
 printf "Suites: $SUITES"
 printf "Components: $COMPONENTS"
 printf "Signed-By: $SIGNATURE"
 [ ! -z "$ARCH_STRING" ] && printf "$ARCH_STRING"
 [ ! -z "$LANG_STRING" ] && printf "$LANG_STRING"
 [ ! -z "$TARGET_STRING" ] && printf "$TARGET_STRING"
 ...
} > /tmp/myconfig.conf 

这是因为如果变量未定义,我不想向文件中写入空的换行符。这种方法的好处是,如果需要,在块内部可以编写各种任意的逻辑。
当然,如果您想要追加内容,您也可以在这里使用>>而不是像其他答案中那样使用>

-1

我通常将模板放在文件中,并使用这个模板引擎:

### <template-file> [ARG=VALUE..]
## Variables are replaced only within "{{" and "}}" notation.
## Example:
##         $0 path-to-tmpl REF=master pass=xx
##         # The template may look like so:
##         #    $pass = ["user", "{{ $pass }}"];
##         # Resulting in:
##         #    $pass = ["user", "xxx"];
##~
template() {
    tmpl=$1
    shift

    for i in $@; do
        declare $i;
    done

    eval "echo \"$(sed -e 's/"/\\"/g' -e 's/\$/\\$/g' -e 's/{{\s*\\\(\$\w*\)\s*}}/\1/g' $tmpl)\""
}

-3

我认为还有一种更简单的方法,但适用于少量代码行。

touch myfile.txt
echo "line1">>myfile.txt
echo "line2">>myfile.txt
echo "line3">>myfile.txt
echo "line4">>myfile.txt

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接