在 Linux 系统中,当你输⼊⼀个命令,再按两次 TAB 键,就会列出所有以你输⼊字符开头的可⽤命令。这并不新鲜,可能你已经知道了。这个功能被称作命令⾏补全bash completion。
默认情况下,bash 命令⾏可以⾃动补全⽂件或⽬录名称。不过,我们可以增强 bash 命令补全功能,通过 complete 命令让它达到新的⾼度。
我们是怎样使⽤可编程的命令⾏补全功能(programmable completion)把⾃动补全功能应⽤于选项或者命令⾏参数。例如:在输⼊ write 命令之后,如果你按两次 TAB 按键,⾃动补全功能会提供可供执⾏ write 操作的⽤户列表。$ write [TAB][TAB]bala rajjason randyjohn ritumayla thomasnisha www-data
在下⾯的例⼦中,可以为 telnet 命令显⽰可⽤的主机名:$ telnet [TAB][TAB]localhost dev-db fileserver
要让可编程命令补全功能在你的终端起作⽤ ,你只需要如下执⾏/etc/bash_completion即可:# . /etc/bash_completion
你也可以取消/etc/bash.bashrc(来⾃ Ubuntu Linux 13.04 系统)中如下的注释,这样,你就可以不需要执⾏上⾯的命令了:### enable bash completion in interactive shells if ! shopt -oq posix; then if [ -f /usr/share/bash-completion/bash_completion ]; then . /usr/share/bash-completion/bash_completion elif [ -f /etc/bash_completion ]; then . /etc/bash_completion fi fi
如果你没有发现这些代码,也没有找到/etc/bash_completion⽂件,那么你只需要通过使⽤apt-get命令来安装bash_completion 包即可。1. 查看已有的命令⾏补全
在启⽤可编程的命令⾏补全功能后,就已经有了⼀些定义好的命令补全功能。complete 命令⽤于定义命令⾏补全。要查看已有的命令⾏补全,如下使⽤ complete 命令:complete -p | less
上⾯例⼦中的 -p 选项是可选的。2. 列出 bash 中标准补全功能
默认情况下,Bash 为 Linux ⽤户提供了下列标准补全功能。
变量补全⽤户名补全主机名补全路径补全⽂件名补全
我们在之前的 中讨论过这些。3. 定义⼀个命令名补全
通过 -c 选项可以将所有的可⽤命令作为⼀个命令的补全参数。在下⾯的例⼦⾥⾯,为 which 命令定义了⼀个补全(LCTT译注:在按两下 TAB 时,可以列出所有命令名作为可补全的参数)。$ complete -c which
$ which [TAB][TAB]Display all 2116 possibilities? (y or n)
如上,如果按下 ‘y’,就会列出所有的命令名。4. 定义⼀个⽬录补全
通过选项 -d,可以定义⼀个仅包含⽬录名的补全参数。在下⾯的例⼦中,为 ls 命令定义了补全。$ ls
countfiles.sh dir1/ dir2/ dir3/
$ complete -d ls
$ ls [TAB][TAB]dir1/ dir2/ dir3/
如上,连按下 TAB 仅会显⽰⽬录名。5. 定义⼀个后台任务名补全
补全功能也能够以任务名作为补全参数。选项 -j 可以定义任务名作为传递给命令的参数,如下:$ jobs[1]- Stopped cat[2]+ Stopped sed 'p'
$ complete -j ./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB]cat sed
关于后台任务,你可以参考 中的例⼦了解如何管理后台任务。6. 带有前缀和后缀的补全
补全功能可以为实际的补全内容定义前缀和后缀。在下⾯的例⼦中,为 list_job_attrib.sh 定义了补全内容的前缀和后缀。$ jobs [1]+ Stopped cat
$ complete -P '\">' -S '<\"' ./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB]
$ ./list_job_attrib.sh \">cat<\"
7. 带有排除的⽂件名和⽬录名补全假如脚本运⾏完成后,输出⽬录如下:$ cd output/ $ ls
all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txtparser_mod.tmp extract.o
如上,如果你想要 ls 命令的补全忽略 .tmp 和 .o ⽂件:$ export FIGNORE='.tmp:.o'
$ complete -f -d ls
$ cd output
$ ls [TAB][TAB]
all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txt
FIGNORE 是⼀个环境变量,它包含了⾃动补全所需要排除的⽂件名后缀。8. 通过 IFS 变量分割字符串得到补全值
可以通过 -W 选项定义补全值列表,然后通过 IFS 环境变量进⾏切分。切分结果会展开变量并作为补全显⽰。$ export IFS=\" \"
$ complete -W \"bubble quick\" ./sort_numbers.sh
$ ./sort_numbers.sh [TAB][TAB]bubble quick
如上所述,字符串通过 IFS 分隔符进⾏切分后,内嵌的变量会被展开为变量值,所以可以如下使⽤变量:echoSORT_TYPE1bubble
echoSORT_TYPE2quick
complete -W \"SORT_TYPE1 $SORT_TYPE2\" ./sort_numbers.sh$ ./sort_numbers.sh [TAB][TAB]bubble quick
9. 写个函数来⽣成补全
你可以引⼊⼀个函数来定义补全。使⽤ -F 选项将函数名传给 complete 命令,执⾏函数⽣成补全内容。例如,函数如下:_parser_options(){ local curr_arg;
curr_arg=${COMP_WORDS[COMP_CWORD]}
COMPREPLY=( (compgen -W '-i --incoming -o --outgoing -m --missed' -- curr_arg ) );}在上述函数中:
COMPREPLY : 该数组控制连按下 TAB 后显⽰的结果COMP_WORDS : 该数组包含命令⾏输⼊的单词
COMP_CWORD : COMP_WORDS 数组的索引,使⽤它来区分命令⾏可以访问的单词位置compgen : -W 基于 $current_arg 提供可能的补全及其参数该函数放在 parser_option ⽂件中,并通过 source 命令引⼊:$ source parser_option
将该函数和你的 parser.pl 脚本关联起来:$ complete -F _parser_options ./parser.pl
$ ./parser.pl [TAB][TAB]-i --incoming -o --outgoing -m --missed如上,parser.pl 的选项是由函数 _parser_options() ⽣成的。
提⽰: 查看/etc/bash_completion 来了解更多的可编程补全函数。10. 当第⼀个规则没有⽣成结果时,就使⽤第⼆个
如果定义的补全规则没有⽣成匹配时,可以使⽤ -o 选项⽣成补全。$ complete -F _count_files -o dirnames ./countfiles.sh
如上,为 ./countfiles.sh 定义了 _count_files 补全函数。 如果 the _count_files() 函数没有⽣成任何匹配的话,就会触发⽬录补全。$ ls
countfiles.sh dir1/ dir2/ dir3/
$./countfiles.sh [TAB][TAB]dir1 dir2 dir3
Loading [MathJax]/jax/element/mml/optable/BasicLatin.js
因篇幅问题不能全部显示,请点此查看更多更全内容