在Linux系統中,輸入一個命令,再按兩次TAB鍵,就會列出所有以輸入字符開頭的可用命令。這并不新鮮,很可能你已經知道了這個。這個功能被稱作命令補全。默認情況下,bash命令行可以自動補全文件或目錄名稱。不過,我們可以使bash命令行補全執行更多的操作,通過補全命令可以讓它成就下一個輝煌。
這個教程說明了我們是怎樣使用可編程的命令行補全功能(programmable completion)把自動補全的功能應用于選項或者命令行參數。
例如:在輸入write 命令之后,如果你按兩次TAB按鍵,自動補全功能會提供執行write操作的列表。
1 2 3 4 5 6 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 包即可。
1、查看現有的bash補全命令
啟用可編程的bash命令行補全功能,就可以定義一套bash補全命令。命令行補全可以用來定義bash補全命令。
來看一下現有的bash補全功能,使用完整的命令,如下:
1 2 | complete -p | less |
選項 -p 是可選擇的。
2、Bash中標準補全的列表
Bash為linux用戶默認提供了下面的標準補全命令。
- 變量名補全(Variablename completion)
- 用戶名補全(Username completion)
- 主機名補全(Hostname completion)
- Path路徑補全(Pathname completion)
- 文件名補全(Filename completion)
我們已經在更早的一篇文章bash standard completion?中討論了這些。
3、為獲取命令定義補全命令
使用-c參數定義一個補全命令來獲得可使用的命令列表。在下面的例子中,為which命令定義了補全命令,
1 2 3 4 5 | $ complete -c which $ which [TAB][TAB] Display all 2116 possibilities? (y or n) |
就像上面看到的,如果按”y”,所有的命令都會顯示出來。
4、為獲得目錄定義補全命令
使用參數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,就可以看到目錄名稱。
5、為獲得后臺作業名稱獲得補全命令
通過使用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 后臺任務。
6、使用前綴和后綴補全命令
補全命令可以通過被前綴(在后面添加)和后綴(添加在后面)來定義。在下面的例子中,前綴和后綴在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<" |
7、具有排除功能的文件名和目錄補全
看看下面的腳本,輸出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變量,它包含了排除在自動補全隊列中的文件的文件名的后綴。
8、通過IFS變量分割String字符串,得到被分割后的值。
單詞表可以通過使用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 |
9、編寫你自己的函數以實現自動補全功能
你可以聲明一個函數來定義補全功能。使用 -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文件,了解更多的可編程的命令行補全功能函數。
10、當第一規范沒有進行匹配時,就需要執行第二規范
通過定義的補全規范,沒有進行匹配,那么通過-o參數定義的completion 就會執行。
1 2 | $ complete -F _count_files -o dirnames ./countfiles.sh?? |
1 2 3 4 5 | $ ls countfiles.sh????dir1/??????dir2/??????dir3/ $./countfiles.sh [TAB][TAB] dir1????dir2????dir3 |