Bash 'here document' 单词/分隔符不加引号



我有以下脚本:

#!/bin/bash
cat << EOF
^A^B^C
EOF

其中:

  • '^a'=十进制值1
  • '^b'=十进制值2
  • '^c'=十进制值3

所以,文件看起来像这样:

xxd main.sh 
00000000: 2321 2f62 696e 2f62 6173 680a 6361 7420  #!/bin/bash.cat 
00000010: 3c3c 2045 4f46 0a01 0203 0a45 4f46 0a    << EOF.....EOF.

当我运行脚本时,十进制值1似乎消失了:

./main.sh | xxd
00000000: 0203 0a                                  ...

我必须引用单词/定界符( cat << "EOF"(才能出现小数值1,但我不明白为什么。

从bash参考手册中,这是当word未引用时发生的事情:

如果 word 均未引用,则此处文档的所有行均受到参数扩展,命令替代和算术扩展的约束,则字符序列 newline被忽略,必须使用''引用字符'','$'和'`'。

我的问题是:

当未引用单词/定界符时,小数值1为什么消失?参数扩展,命令替代和算术扩展如何影响这?

正如评论中指出的那样,我相信这也是一个错误。

在我的示例中,BASH最终将此处的文档文本转换为文件,而这样做的过程会将此处的文档文本转换为某种内部字符串(ISTRING(。这是我执行的回溯:

#0  expand_word_internal (word=0x7fffffffd610, quoted=2, isexp=0, contains_dollar_at=0x0, expanded_something=0x0) at subst.c:9155
#1  0x00005555555c1bb9 in call_expand_word_internal (w=0x7fffffffd610, q=2, i=0, c=0x0, e=0x0) at subst.c:3614
#2  0x00005555555c1cb3 in expand_string_internal (string=0x5555558a6b28 "010203n", quoted=2) at subst.c:3649
#3  0x00005555555c2088 in expand_string_leave_quoted (string=0x5555558a6b28 "010203n", quoted=2) at subst.c:3777
#4  0x00005555555c2197 in expand_string (string=0x5555558a6b28 "010203n", quoted=2) at subst.c:3825
#5  0x00005555555f203b in write_here_document (fd=3, redirectee=0x5555558a6b08) at redir.c:394
#6  0x00005555555f227f in here_document_to_fd (redirectee=0x5555558a6b08, ri=r_reading_until) at redir.c:476
#7  0x00005555555f2f8b in do_redirection_internal (redirect=0x5555558a6b48, flags=1) at redir.c:970
#8  0x00005555555f1cba in do_redirections (list=0x5555558a6b48, flags=1) at redir.c:234
#9  0x00005555555a347d in execute_disk_command (words=0x5555558a5b08, redirects=0x5555558a6b48, command_line=0x5555558a5408 "cat  <<    EOFn010203nEOFn", pipe_in=-1, pipe_out=-1, async=0,
    fds_to_close=0x5555558a6b88, cmdflags=0) at execute_cmd.c:5212
#10 0x00005555555a1ced in execute_simple_command (simple_command=0x5555558a6ac8, pipe_in=-1, pipe_out=-1, async=0,     fds_to_close=0x5555558a6b88) at execute_cmd.c:4386
#11 0x000055555559b098 in execute_command_internal (command=0x5555558a6a88, asynchronous=0, pipe_in=-1, pipe_out=-1,     fds_to_close=0x5555558a6b88) at execute_cmd.c:802
#12 0x000055555559a6a6 in execute_command (command=0x5555558a6a88) at execute_cmd.c:405
#13 0x00005555555841ff in reader_loop () at eval.c:180
#14 0x0000555555581bc3 in main (argc=2, argv=0x7fffffffdb68, env=0x7fffffffdb80) at shell.c:792

expand_word_internal将我的字符串从"010203"转换为其内部字符串表示" 01020103"代替"010101020103",我相信这是所需的行为。

最新更新