在 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 shellsif ! 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 fifi
如果你沒有發現這些代碼,也沒有找到/etc/bash_completion檔案,那麼你只需要通過使用apt-get命令來安裝bash_completion 包即可。
1. 查看已有的命令列補全
在啟用可程式化的命令列補全功能後,就已經有了一些定義好的命令補全功能。complete 命令用於定義命令列補全。
要查看已有的命令列補全,如下使用 complete 命令:
complete -p | less
上面例子中的 -p 選項是可選的。
2. 列出 bash 中標準補全功能
預設情況下,Bash 為 Linux 使用者提供了下列標準補全功能。
1.變數補全
2.使用者名稱補全
3.主機名稱補全
4.路徑補全
5.檔案名稱補全
我們在之前的 bash 標準補全中討論過這些。
3. 定義一個命令名補全
通過 -c 選項可以將所有的可用命令作為一個命令的補全參數。在下面的例子裡面,為 which 命令定義了一個補全(LCTT譯註:在按兩下 TAB 時,可以列出所有命令名作為可補全的參數)。
$ complete -c which$ which [TAB][TAB]Display all 2116 possibilities? (y or n)
如上,如果按下 ‘y',就會列出所有的命令名。
4. 定義一個目錄補全
通過選項 -d,可以定義一個僅包含目錄名的補全參數。在下面的例子中,為 ls 命令定義了補全。
$ lscountfiles.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
關於背景工作,你可以參考 Linux 背景工作中的例子瞭解如何管理背景工作。
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/$ lsall_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 分隔字元進行切分後,內嵌的變數會被展開為變數值,所以可以如下使用變數:
$ echo $SORT_TYPE1bubble$ echo $SORT_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 ) );}
在上述函數中:
1.COMPREPLY : 該數組控制連按下 TAB 後顯示的結果
2.COMP_WORDS : 該數組包含命令列輸入的單詞
3.COMP_CWORD : COMP_WORDS 數組的索引,使用它來區分命令列可以訪問的單詞位置
4.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
以上就是本文的全部內容,希望對大家的學習有所協助,也希望大家多多支援雲棲社區。