在Linux系統中,輸入一個命令,再按兩次TAB鍵,就會列出所有以輸入字符開頭的可用命令。這並不新鮮,很可能你已經知道了這個。這個功能被稱作命令補全。默認情況下,bash命令行可以自動補全文件或目錄名稱。不過,我們可以使bash命令行補全執行更多的操作,通過補全命令可以讓它成就下一個輝煌。
這個教程說明了我們是怎樣使用可編程的命令行補全功能(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 包即可。
啟用可編程的bash命令行補全功能,就可以定義一套bash補全命令。命令行補全可以用來定義bash補全命令。
來看一下現有的bash補全功能,使用完整的命令,如下:
complete -p | less
選項 -p 是可選擇的。
Bash為linux用戶默認提供了下面的標准補全命令。
我們已經在更早的一篇文章bash standard completion 中討論了這些。
使用-c參數定義一個補全命令來獲得可使用的命令列表。在下面的例子中,為which命令定義了補全命令,
$ 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,就可以看到目錄名稱。
通過使用complete命令,把獲得job名稱作為參數是允許的。參數j用來把job名稱作為參數傳到命令行,展示如下:
$ 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中被定義。
$ jobs [1]+ Stopped cat $ complete -P '">' -S '<"' ./list_job_attrib.sh $ ./list_job_attrib.sh [TAB][TAB] $ ./list_job_attrib.sh ">cat<"
看看下面的腳本,輸出output 目錄下面的文件:
$ cd output/ $ ls all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txt parser_mod.tmp extract.o
在上面的例子中,如果你想要排除以.tmp和.o為後綴的文件,實現ls命令的自動補全功能,可以這樣:
$ 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 變量中定義的字符串分割成多個單詞。最終每個單詞都會被分開,被顯示出來。
$ 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 參數,被傳入到補全命令的函數名,可以執行並。例如,函數可以寫成下面的樣式。
_parser_options() { local curr_arg; curr_arg=${COMP_WORDS[COMP_CWORD]} COMPREPLY=( $(compgen -W '-i --incoming -o --outgoing -m --missed' -- $curr_arg ) ); }
在上面的函數中,
文件中parser_option 函數通過source執行如下:
$ source parser_option
這個函數鏈接到腳本解析器如下:
$ complete -F _parser_options ./parser.pl $ ./parser.pl [TAB][TAB] -i --incoming -o --outgoing -m --missed
就像上面所看到的,解析器的參數可以通過_parser_options函數生成。
注意:查看/etc/bash_completion文件,了解更多的可編程的命令行補全功能函數。
通過定義的補全規范,沒有進行匹配,那麼通過-o參數定義的completion 就會執行。
$ complete -F _count_files -o dirnames ./countfiles.sh
同上,通過使用_count_files 文件中定義的_count_files 函數定義的completion ,如果_count_files函數沒有進行匹配,那麼就會執行目錄補全。
$ ls countfiles.sh dir1/ dir2/ dir3/ $./countfiles.sh [TAB][TAB] dir1 dir2 dir3
原文鏈接: Balakrishnan Mariyappan 翻譯: 極客范 - 踏雁尋花