3

前几天我正在尝试使用 shell 函数,目的是覆盖ls命令。目的是让shell首先调用我的ls函数,然后如果第一个开关是“-y”做一件事,否则调用常规/bin/ls,但结果的行为方式我不明白。

为了启用它,我决定使用“-y”作为开关,因为:

$ ls -y
ls: invalid option -- 'y'
Try `ls --help' for more information.

所以它不能破坏ls. 无论如何,将问题简化为最简单的问题,并使用其他几个示例来帮助突出问题:

$ function ls() { [[ "y$1" == "y-y" ]] && echo LS ; } 
$ function less() { [[ "y$1" == "y-y" ]] && echo LESS ; }
$ function try() { [[ "y$1" == "y-y" ]] && echo TRY ; }

所以我要压倒一切lsless并定义try哪个就像“控制”标本:)

现在以这种方式调用:

$ ls ; less ; try 

它的行为符合预期(无输出),但是:

$ ls -y ; less -y ; try -y
LESS
TRY

ls无法工作,但less覆盖确实如此,“控制”也是如此。

在 stackoverflow 或 askubuntu 的其他地方(但我暂时失去了参考)我看到了一个暗示ls是 builltin 的例子,但是:

$ which ls
/bin/ls

和:

$ builtin ls
bash: builtin: ls: not a shell builtin

所以它似乎不在我的系统上(即使它是我实际上也看不出它为什么会这样)。

我想知道解释是什么?这并不重要,但我想了解发生了什么。

谢谢。

4

3 回答 3

6

不使用which,使用type。如果你有类似的东西

$ type ls
ls is aliased to `ls --color=auto'

然后您的函数将在交互式 shell 中失败。.bashrc在您或任何定义的地方删除它。你也可以使用unalias内置的。

如果在函数中运行与函数同名的命令,还需要避免递归。

ls()
    if [[ $1 == -y ]]; then
        shift
        ...
    else
        command ls "$@"
    fi

function此外,如果您决定使用关键字定义函数,请确保您知道自己在做什么。funcname() compound-command在撰写本文时,如果您需要 POSIX 兼容性,请使用 POSIX 样式。

于 2012-06-30T07:18:24.940 回答
5

为 设置别名很常见,ls您可能就是这种情况。首先处理别名,这是简单的文本替换。您的示例中发生的情况是:

A) `ls` is replaced with `ls --color=auto` 
B) Your `ls` function is called with the replaced text and your -y option 
C) Your function checks "$1", which is --color=auto

下面显示了此行为的示例:

$ function ls() { [[ "y$1" == "y-y" ]] && echo LS ; }
$ ls -y
$ type ls
ls is aliased to `ls --color=auto'
$ unalias ls
$ type ls
ls is a function
ls ()
{
    [[ "y$1" == "y-y" ]] && echo LS
}
$ function ls() { [[ "y$1" == "y-y" ]] && echo LS ; }
$ ls -y
LS
$ alias ls="ls --color=auto"
$ ls -y
--color=auto -y
于 2012-06-30T07:16:23.507 回答
1

我猜是

$ alias ls

会告诉你原因。

于 2012-06-30T07:17:04.363 回答