bash别名即使使用shopt expand_aliases也不会扩展


8

我想在bash -c构造内运行别名。

bash手册说:

如果外壳不是交互式的,则别名不会扩展,除非expand_aliases使用以下命令设置外壳选项shopt

在此示例中,为什么hiexpand_aliases显式设置时找不到别名?

% bash -O expand_aliases -c "alias hi='echo hello'; alias; shopt expand_aliases; hi"
alias hi='echo hello'
expand_aliases  on
bash: hi: command not found

我在跑步GNU bash, version 5.0.0(1)-release (x86_64-pc-linux-gnu)

上下文:我希望能够以空闲优先级运行别名,例如包含以下内容的脚本:

#!/bin/bash
exec chrt -i 0 nice -n 19 ionice -c 3 bash -c ". ~/.config/bash/aliases; shopt -s expand_aliases; $(shell-quote "$@")"

我不想使用它,bash -i因为我不想.bashrc被阅读。


3
Bash手册中带引号的语句之后的段落似乎涵盖了以下内容:'...别名在读取命令时展开,而不是在执行命令时展开。因此,与另一命令出现在同一行上的别名定义在读取下一行输入之前不会生效。该行上别名定义之后的命令不受新别名的影响
。...'– Haxiel

与大多数情况一样,您应该在此处考虑使用shell函数而不是别名。bash -c "hi () { echo hello; }; hi"输出hello
chepner '19

Answers:


16

如果在使用别名的同一行上设置别名,这似乎不起作用。可能与别名在实际解析阶段之前的命令行处理过程中的早期扩展有关。在交互式外壳上:

$ alias foo
bash: alias: foo: not found
$ alias foo='echo foo'; foo         # 2 
bash: foo: command not found
$ alias foo='echo bar'; foo         # 3
foo
$ foo
bar

请注意,别名使用的时间晚了一行:在第二个命令中,它找不到刚刚设置的别名,而在第三个命令中,它使用了先前设置的别名。

因此,如果我们在-c字符串中放置换行符,它将起作用:

$ bash -c $'shopt -s expand_aliases; alias foo="echo foo";\n foo'
foo

(您也可以在脚本中使用bash -O expand_aliases -c ...而不是使用shopt它,但这并不是换行符的帮助。)

另外,您可以使用shell函数代替别名,它们在其他方面也要好得多:

$ bash -c 'foo() { echo foo; }; foo'
foo

14

根据ilkkachu的建议,将我的评论转化为答案。

Bash手册(在问题中链接到)确实解释了在同一行上有别名定义和命令时如何处理别名。

引用(为清晰起见,略有格式):

有关别名的定义和使用的规则有些混乱。在执行该行或复合命令上的任何命令之前,Bash始终至少读取完整的一行输入以及构成复合命令的所有行。

读取命令时(而不是执行命令时),别名会扩展。因此,与另一条命令在同一行上出现的别名定义在读取下一行输入之前不会生效。该行中别名定义之后的命令不受新别名的影响。

执行功能时,此行为也是一个问题。读取函数定义时(而不是执行函数时)会扩展别名,因为函数定义本身就是命令。结果,在函数中定义的别名直到该函数执行后才可用。

为了安全起见,请始终将别名定义放在单独的行上,并且不要在复合命令中使用别名。

ilkkachu的答案为该问题提供了多种可能的解决方案。


FWIW,我看到了您的最后评论,但没有时间回答。对于答案来补充其他人来说不是一件坏事,并且知道它实际上已被记录下来很有用的。因此,感谢您撰写本文,现在我可以对其进行投票。:)
ilkkachu
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.