在 Linux 系統中,當你輸入一個命令,再按兩次 TAB
鍵,就會列出所有以你輸入字符開頭的可用命令。這並不新鮮,可能你已經知道了。這個功能被稱作命令行補全bash completion。默認情況下,bash 命令行可以自動補全文件或目錄名稱。不過,我們可以增強 bash 命令補全功能,通過 complete
命令讓它達到新的高度。
這個教程說明了我們是怎樣使用可編程的命令行補全功能(programmable completion)把自動補全功能應用於選項或者命令行參數。
例如:在輸入 write
命令之後,如果你按兩次 TAB
按鍵,自動補全功能會提供可供執行 write
操作的用戶列表。
$ write[TAB][TAB]
bala raj
jason randy
john ritu
mayla thomas
nisha 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 包即可。
在啟用可編程的命令行補全功能後,就已經有了一些定義好的命令補全功能。complete
命令用於定義命令行補全。
要查看已有的命令行補全,如下使用 complete
命令:
complete -p |less
上面例子中的 -p 選項是可選的。
默認情況下,Bash 為 Linux 用戶提供了下列標准補全功能。
我們在之前的 bash 標准補全中討論過這些。
通過 -c 選項可以將所有的可用命令作為一個命令的補全參數。在下面的例子裡面,為 which
命令定義了一個補全(LCTT譯注:在按兩下 TAB
時,可以列出所有命令名作為可補全的參數)。
$ complete -c which
$ which [TAB][TAB]
Display all 2116 possibilities?(y or n)
如上,如果按下 ‘y’,就會列出所有的命令名。
通過選項 -d,可以定義一個僅包含目錄名的補全參數。在下面的例子中,為 ls
命令定義了補全。
$ ls
countfiles.sh dir1/ dir2/ dir3/
$ complete -d ls
$ ls[TAB][TAB]
dir1/ dir2/ dir3/
如上,連按下 TAB
僅會顯示目錄名。
補全功能也能夠以任務名作為補全參數。選項 -j 可以定義任務名作為傳遞給命令的參數,如下:
$ jobs
[1]-Stoppedcat
[2]+Stoppedsed'p'
$ complete -j ./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB]
catsed
關於後台任務,你可以參考 Linux 後台任務中的例子了解如何管理後台任務。
補全功能可以為實際的補全內容定義前綴和後綴。在下面的例子中,為 list_job_attrib.sh 定義了補全內容的前綴和後綴。
$ jobs
[1]+Stoppedcat
$ complete -P '">'-S '<"'./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB]
$ ./list_job_attrib.sh ">cat<"
假如腳本運行完成後,輸出目錄如下:
$ cd output/
$ ls
all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txt
parser_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
是一個環境變量,它包含了自動補全所需要排除的文件名後綴。
可以通過 -W 選項定義補全值列表,然後通過 IFS
環境變量進行切分。切分結果會展開變量並作為補全顯示。
$ export IFS=" "
$ complete -W "bubble quick"./sort_numbers.sh
$ ./sort_numbers.sh [TAB][TAB]
bubble quick
如上所述,字符串通過 IFS
分隔符進行切分後,內嵌的變量會被展開為變量值,所以可以如下使用變量:
$ 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 選項將函數名傳給 complete
命令,執行函數生成補全內容。例如,函數如下:
_parser_options()
{
local curr_arg;
curr_arg=${COMP_WORDS[COMP_CWORD]}
COMPREPLY=( $(compgen -W '-i --incoming -o --outgoing -m --missed'-- $curr_arg ));
}
在上述函數中:
TAB
後顯示的結果該函數放在 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
來了解更多的可編程補全函數。
如果定義的補全規則沒有生成匹配時,可以使用 -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