Bash完成:我们可以用它做什么,未来是什么

时间:2009-12-13 13:10:54

标签: bash tab-completion programmable-completion

Bash允许您使用TAB键完成参数中的命令名称和文件名。 但为什么不是命令的常见选项呢?为什么不是,甚至更好的是,一个完成系统会告诉你一个选项的作用呢?

我听说可编程完成..但不明白它适合的地方..

所以我的问题是:有没有办法实现我的要求?与Bash结合的其他工具可能......或者其他什么?

7 个答案:

答案 0 :(得分:5)

如果您想要更加精彩的完成,请查看zsh。它非常像bash,并且从ksh,tcsh等中获取了一些很好的功能。它还修复了一些烦人的bash bug。

它可以做一些非常酷的东西,包括为scp完成远程文件名(如果你有基于密钥的身份验证设置)和完成选项卡时的迷你文档 例如,当我前几天在“jar”上标记时,我感到惊喜:

[steven@sexy:~]% jar <tab>
0  -- store only without using ZIP compression
M  -- do not create manifest file
etc...

答案 1 :(得分:2)

以下是bash的官方手册页:Bash Reference Manual: Programmable Completion

答案 2 :(得分:2)

在OS X上,您可以按照here说明使用MacPorts轻松安装一组预定义的bash完成。查看/opt/local/etc/bash_completion.d中的示例可能会为您提供一些有关制作其他示例的建议。

答案 3 :(得分:1)

这是bash的标准功能:programmable completion

答案 4 :(得分:1)

Bash确实支持参数完成:http://www.debian-administration.org/articles/316

答案 5 :(得分:1)

Bash完成是非常可定制的,可以显示常用命令的选项。配置文件只需要指定即可。在Debian中,bash完成带有一组配置文件,可以完成不同命令的选项。

例如:

alanhaggai@neon:~$ tar
A  c  d  r  t  u  x

答案 6 :(得分:0)

我不喜欢这里的任何答案,并希望提供一些具体的例子。

我有时会有一些带有公共前缀的目录,并希望能够通过输入目录名称的唯一部分(位于目录名称的中间或末尾)来完成它们。

我能够写一个完成函数来完成这个:

_cd(){
    # Backup old nullglob setting
    local shoptbakup="`shopt -p nullglob`"
    shopt -s nullglob

    local cur opts i opt
    local IFS=$'\n'
    cur="${COMP_WORDS[COMP_CWORD]}"

    # Home directory substitution
    if [[ "${cur:0:1}" == '~' ]]; then
        cur="$HOME/${cur:1}"
    fi

    if [[ $cur == */* ]]; then
        opts=(${cur}*/)
    else
        opts=(*${cur}*/)
    fi

    # Prevent trailing//
    i=0
    for opt in "${opts[@]}"; do
        #opts[$i]=${opt%${opt##*[!/]}}
        opts[$i]=${opt%/}
        i=$((i+1))
    done

    # Restore nullglob setting
    eval "$shoptbakup" 2>/dev/null

    COMPREPLY=("${opts[@]}")
}
complete -o filenames -o bashdefault -o nospace -F _cd cd

所以现在从我的提示我可以这样做:

$ ls
test 6/  test 7/  test 8/  test1/  test2/  test3/  test4/  test5/
$ cd 7[TAB]

它会自动完成:

$ cd test\ 7/

您可能希望使用可编程完成的另一个原因是,如果您编写脚本并希望能够为您列出或完成选项。这个页面有一些很好的例子:

http://fahdshariff.blogspot.com/2011/04/writing-your-own-bash-completion.html

以下是一个例子:

假设我有一个名为listanimals的脚本。我可以写一个完成函数来帮助填写选项

# complete sendevent
_listanimals()
{
    local cur=${COMP_WORDS[COMP_CWORD]}
    local prev=${COMP_WORDS[COMP_CWORD-1]}

    if [[ $prev == "--sort-direction" ]]; then
        COMPREPLY=( $( compgen -W "horizontal vertical" -- $cur ) )
        return 0
    elif [[ $prev == "--type" ]];then
        COMPREPLY=( $( compgen -W "mammals reptiles birds fish marsupials insects misc" -- $cur ) )
        return 0
    elif [[ $prev == "--size" ]]; then
        COMPREPLY=( $( compgen -W "tiny small medium large huge" -- $cur ) )
        return 0
    fi


    # completing an option
    if [[ "$cur" == --* ]] || [[ $cur == "" ]]; then
        COMPREPLY=( $( compgen -W "--sort-direction --type --size" -- $cur ) )
    fi
}
complete -F _listanimals listanimals

现在我可以这样做:

$ listanimals --[TAB][TAB]
--size            --sort-direction  --type
$ listanimals --t[TAB]

将自动填写此内容:

$ listanimals --type

然后我可以这样做:

$ listanimals --type [TAB][TAB]
birds       fish        insects     mammals     marsupials  misc        reptiles

我相信你可以弄明白其余的。这是一个工作示例,因此如果您想尝试将其复制/粘贴到文件(例如sample.sh)中,并将其以bash source sample.sh来源。您实际上不需要一个名为listanimals的脚本。

希望有人会觉得这很有用,因为我觉得很难理解。