我想在bash -c
构造中运行别名。
该bash
手册说:
当 shell 不是交互式时,别名不会被扩展,除非
expand_aliases
shell 选项是使用设置的shopt
在这个例子中,为什么显式hi
设置时找不到别名expand_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
Run Code Online (Sandbox Code Playgroud)
我在跑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 "$@")"
Run Code Online (Sandbox Code Playgroud)
我想避免使用,bash -i
因为我不想.bashrc
被阅读。
ilk*_*chu 19
如果在使用的同一行上设置别名,它似乎不起作用。可能与如何在命令行处理的早期,在实际解析阶段之前真正扩展别名有关。在交互式外壳上:
$ 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
Run Code Online (Sandbox Code Playgroud)
请注意使用的别名是如何延迟一行的:在第二个命令中,它没有找到刚刚设置的别名,而在第三个命令中,它使用先前设置的别名。
因此,如果我们在-c
字符串中放置一个换行符,它会起作用:
$ bash -c $'shopt -s expand_aliases; alias foo="echo foo";\n foo'
foo
Run Code Online (Sandbox Code Playgroud)
(您也可以使用bash -O expand_aliases -c ...
而不是shopt
在脚本中使用,而不是它有助于换行。)
或者,您可以使用 shell 函数而不是别名,它们在其他方面也更好:
$ bash -c 'foo() { echo foo; }; foo'
foo
Run Code Online (Sandbox Code Playgroud)
Hax*_*iel 14
正如 ilkkachu 所建议的那样,将我的评论变成答案。
Bash 手册(在问题中链接到)确实提供了在同一行上有别名定义和命令时如何处理别名的说明。
报价(为清晰起见略有格式化):
有关别名的定义和使用的规则有些令人困惑。在执行该行上的任何命令或复合命令之前,Bash 总是至少读取一个完整的输入行,以及构成复合命令的所有行。
别名在读取命令时展开,而不是在执行时展开。因此,与另一个命令出现在同一行上的别名定义在读取下一行输入之前不会生效。该行别名定义后面的命令不受新别名的影响。
执行函数时,此行为也是一个问题。别名在读取函数定义时展开,而不是在执行函数时展开,因为函数定义本身就是一个命令。因此,函数中定义的别名在执行该函数之前不可用。
为安全起见,始终将别名定义放在单独的行上,并且不要在复合命令中使用别名。
ilkkachu 的回答为这个问题提供了多种可能的解决方案。