在Linux系统中,输入一个命令,再按两次TAB键,就会列出全部以输入字符开头的可用命令。这并不新鲜,极可能你已经知道了这个。这个功能被称做命令补全。默认状况下,bash命令行能够自动补全文件或目录名称。不过,咱们能够使bash命令行补全执行更多的操做,经过补全命令可让它成就下一个辉煌。linux
这个教程说明了咱们是怎样使用可编程的命令行补全功能(programmable completion)把自动补全的功能应用于选项或者命令行参数。shell
例如:在输入write 命令以后,若是你按两次TAB按键,自动补全功能会提供执行write操做的列表。编程
1数组 2bash 3less 4函数 5ui 6spa 7命令行 |
$ write [TAB][TAB] bala raj jason randy john ritu mayla thomas nisha www-data
|
在下面的例子中,输入telnet命令将会显示可用了主机名:
1 2 3 |
$ telnet [TAB][TAB] localhost dev-db fileserver
|
要让可编程命令补全功能在你的终端起做用 ,你只须要执行/etc/bash_completion便可,就像下面展现出来的操做:
1 2 |
# . /etc/bash_completion
|
你也能够取消/etc/bash.bashrc(对于Ubuntu Linux 13.04系统)下面的注释,这样,你就能够不须要执行上面的命令了,
1 2 3 4 5 6 7 8 9 |
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 包便可。
启用可编程的bash命令行补全功能,就能够定义一套bash补全命令。命令行补全能够用来定义bash补全命令。
来看一下现有的bash补全功能,使用完整的命令,以下:
1 2 |
complete -p | less
|
选项 -p 是可选择的。
Bash为linux用户默认提供了下面的标准补全命令。
变量名补全(Variablename completion)
用户名补全(Username completion)
主机名补全(Hostname completion)
Path路径补全(Pathname completion)
文件名补全(Filename completion)
咱们已经在更早的一篇文章bash standard completion 中讨论了这些。
使用-c参数定义一个补全命令来得到可以使用的命令列表。在下面的例子中,为which命令定义了补全命令,
1 2 3 4 5 |
$ complete -c which
$ which [TAB][TAB] Display all 2116 possibilities? (y or n)
|
就像上面看到的,若是按”y”,全部的命令都会显示出来。
使用参数d,定义一个只得到目录名称的补全命令,下面的例子中,定义了ls的补全命令
1 2 3 4 5 6 7 8 |
$ ls countfiles.sh dir1/ dir2/ dir3/
$ complete -d ls
$ ls [TAB][TAB] dir1/ dir2/ dir3/
|
就像上面看到的,连续按两次TAB,就能够看到目录名称。
经过使用complete命令,把得到job名称做为参数是容许的。参数j用来把job名称做为参数传到命令行,展现以下:
1 2 3 4 5 6 7 8 9 |
$ jobs [1]- Stopped cat [2]+ Stopped sed 'p'
$ complete -j ./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB] cat sed
|
想要了解更多的后台任务,能够经过这些案例来了解下如何管理Linux 后台任务。
补全命令能够经过被前缀(在后面添加)和后缀(添加在后面)来定义。在下面的例子中,前缀和后缀在list_job_attrib.sh中被定义。
1 2 3 4 5 6 7 8 9 |
$ jobs [1]+ Stopped cat
$ complete -P '">' -S '<"' ./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB]
$ ./list_job_attrib.sh ">cat<"
|
看看下面的脚本,输出output 目录下面的文件:
1 2 3 4 5 6 |
$ cd output/
$ ls all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txt parser_mod.tmp extract.o
|
在上面的例子中,若是你想要排除以.tmp和.o为后缀的文件,实现ls命令的自动补全功能,能够这样:
1 2 3 4 5 6 7 8 9 |
$ 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 是一个shell变量,它包含了排除在自动补全队列中的文件的文件名的后缀。
单词表能够经过使用w参数被IFS 变量中定义的字符串分割成多个单词。最终每一个单词都会被分开,被显示出来。
1 2 3 4 5 6 7 |
$ export IFS=" "
$ complete -W "bubble quick" ./sort_numbers.sh
$ ./sort_numbers.sh [TAB][TAB] bubble quick
|
如上所述,被IFS分割以后,单词就会被扩展开,因此也可能有下面展现的这些变量。
1 2 3 4 5 6 7 8 9 10 |
$ echo $SORT_TYPE1 bubble
$ echo $SORT_TYPE2 quick
$ complete -W "$SORT_TYPE1 $SORT_TYPE2" ./sort_numbers.sh $ ./sort_numbers.sh [TAB][TAB] bubble quick
|
你能够声明一个函数来定义补全功能。使用 -F 参数,被传入到补全命令的函数名,能够执行并。例如,函数能够写成下面的样式。
1 2 3 4 5 6 7 8 9 |
_parser_options() { local curr_arg;
curr_arg=${COMP_WORDS[COMP_CWORD]}
COMPREPLY=( $(compgen -W '-i --incoming -o --outgoing -m --missed' -- $curr_arg ) ); }
|
在上面的函数中,
COMPREPLY :存储在按下[TAB][TAB]以后打印信息的数组。
COMP_WORDS :在命令行输入的单词数组
COMP_CWORD :COMP_WORDS 数组的索引,能够访问命令行中不用位置的单词。
compgen :使用-W参数,持有current_arg变量中尽量完整的、分开的内容。
文件中parser_option 函数经过source执行以下:
1 2 |
$ source parser_option
|
这个函数连接到脚本解析器以下:
1 2 3 4 5 |
$ complete -F _parser_options ./parser.pl
$ ./parser.pl [TAB][TAB] -i --incoming -o --outgoing -m --missed
|
就像上面所看到的,解析器的参数能够经过_parser_options函数生成。
注意:查看/etc/bash_completion文件,了解更多的可编程的命令行补全功能函数。
经过定义的补全规范,没有进行匹配,那么经过-o参数定义的completion 就会执行。
1 2 |
$ complete -F _count_files -o dirnames ./countfiles.sh
|
同上,经过使用_count_files 文件中定义的_count_files 函数定义的completion ,若是_count_files函数没有进行匹配,那么就会执行目录补全。
1 2 3 4 5 |
$ ls countfiles.sh dir1/ dir2/ dir3/
$./countfiles.sh [TAB][TAB] dir1 dir2 dir3 |