在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 |