我正在尝试在脚本中创建一个别名:
#!/bin/bash
shopt -s expand_aliases
#Currently just using 'git' to try and get it working
#It'll be replaced with the following line when it works
#compgen -c | while read x;
echo 'git' | while read x;
do
a=`echo $x | tr '[:lower:]' '[:upper:]'`;
echo $a;
echo $x;
if [ "$a" != '' ]
then
echo 'got here';
alias $a=$x;
alias;
GIT;
fi
done
虽然一个简单的测试脚本可以正常工作(testme
被添加到我当前shell中的别名中):
#!/bin/bash
shopt -s expand_aliases
alias testme="echo It Worked"
alias
testme
第一个脚本的输出:
GIT
git
got here
alias GIT='git'
alias grep='grep --colour=auto'
alias ls='ls --color=auto'
GIT: command not found
GIT
显示在别名中,但无法执行。我尝试使用. ./script.sh
和source script.sh
进行调用。我做错了什么?
答案 0 :(得分:7)
当别名扩展时,可能会出现问题。 bash手册页有以下内容:
关于别名的定义和使用的规则有所不同 混乱。在执行该行上的任何命令之前,Bash始终会读取至少一个完整的输入行。在命令时扩展别名 读取,而不是在执行时。 ...
手册页中有更多信息,但是当执行复合语句(如while
循环中的列表)时,扩展将在首次读取输入时发生,而不是在执行时发生。由于在读取输入时未定义GIT
,因此将出现错误。
以下代码段显示了该问题:
#!/bin/bash
shopt -s expand_aliases
while read x; do
a=`echo $x | tr '[:lower:]' '[:upper:]'`
alias $a=$x
GIT # <--- this fails to execute git
done < <(echo 'git')
GIT # <--- this executes git
执行while循环后,别名变为可用。手册页中的详细信息有点粗略,但似乎适用于函数中定义的别名注释。
另外,使用echo foo | while read x; do stuff; done
等语法时要小心。 while
循环将在子shell中执行,子shell中所做的任何更改都不会超出while
循环。上面的示例显示了一个方法,以便while循环在当前shell中执行,但从执行的命令中获取其输入。
<强>更新强>
如果上面的脚本是以原始形式编写的,那么它将如下所示:
#!/bin/bash
shopt -s expand_aliases
#Currently just using 'git' to try and get it working
#It'll be replaced with the following line when it works
#compgen -c | while read x;
echo 'git' | while read x; do
a=`echo $x | tr '[:lower:]' '[:upper:]'`
alias $a=$x
GIT # <--- this fails to execute git
done
GIT # <--- this also fails executes git because the alias' were set in a subshell.
在这种情况下,别名永远不会执行。这样做的原因是管道|
为一个新的子shell加星并将这些过程链接在一起。 echo
在一个进程中运行,while read ...
在另一个进程中运行,由管道链接。设置别名后,它们将在运行while
循环的子shell的进程环境中设置。但是,当while
循环结束时,该进程将终止并丢弃其环境,以及对其环境所做的任何更改,例如添加别名。因此,执行GIT
的第二次尝试将失败,因为别名信息已被丢弃。
当您使用第一个脚本中显示的拓扑时,while
循环在当前脚本中运行,因此对环境所做的任何更改都会持续到循环结束。
如果你执行你的脚本而不是采购它,你也会注意到这种行为。如果您希望能够在这样的脚本中配置一组别名,运行脚本,让它们在您的环境中可用,您遇到同样的问题,因为您的脚本在一个单独的进程中运行到父shell,创建别名,当脚本退出时随后被丢弃。
$ ./create_aliases
$ alias # <--- Ooops, no aliases defined.
您必须像这样来源脚本,它将在当前进程中运行脚本的内容,从而修改当前环境。
$ source ./create_aliases
$ alias # <--- Yay, new aliases present.
这就是为什么bash将提供~/.bashrc
等文件的原因,以便您在该文件中所做的更改将存在于当前环境中。