在bash脚本中设置别名

时间:2013-04-10 02:03:28

标签: bash

我正在尝试在脚本中创建一个别名:

#!/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.shsource script.sh进行调用。我做错了什么?

1 个答案:

答案 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等文件的原因,以便您在该文件中所做的更改将存在于当前环境中。