Friendly Interactive Shell(fish)很容易使用。它的語法、上下文相關的幫助和顏色編碼的命令行界面(CLI)顯著簡化了 UNIX® 的使用,減輕了腳本編程的負擔。
英語是一種令人困惑的語言。例如,請考慮 moon 和 good 這兩個單詞。對外行人而言,這兩個單詞似乎應該是押韻的,但是前者的讀音是 /mun/(根據 International Pronunciation Alphabet),而後者的讀音是 /good/。似乎英語中的惟一規則就是例外。
UNIX shell 同樣令人困惑。例如,在 Bourne shell(和大多數常用的 UNIX shell)中,'$var'、"$var" 和 `$var` 看起來相似,但是它們會產生很不一樣的結果。(在本文中的 shell 示例中,每個 CLI 前面都加上使用的 shell 的名稱和命令編號)。
bash-1) # Demonstrate the differences between single-, double-, and back quotes bash-2) var=ls bash-3) echo '$var' $var bash-4) echo "$var" ls bash-5) echo `$var` Rakefile app bin components config db doc lib log patches public script src test tmp vendor |
在上面的命令序列中,把變數 var 設置為兩字母的字元串 ls。在第一個 echo 命令中,單引號禁止解釋此變數,因此會按原樣顯示引號中的文本,即四字母的字元串 $var。在第 4 行代碼中,雙引號會解釋此變數,所以結果是字元串 ls。最後,反撇號解釋變數並作為子 shell 運行中間結果。因此,`$var` 產生中間字元串 ls,它作為 shell 命令運行,生成當前目錄的內容列表。
當然,這三種操作符(單引號、雙引號和反撇號)都有合法的用途,但是與英語中的例外一樣,記住和掌握這些細微差異很令人頭疼。為了進一步證明這一點,請問:$var 和 "$var" 之間有什麼差異?(提示:假設 $var 包含空格。)
bash-1) # Create three files and try to remove two bash-2) touch three two one bash-3) var="one two" bash-4) rm "$var" rm: one two: No such file or directory bash-5) rm $var bash-6) ls three |
如果一個變數包含空格,雙引號會按原樣把變數展開為一個 參數。否則,變數中的任何空格都被解釋為參數分隔符。
shell 語法很令人頭疼。這很糟糕,因為它使 CLI(UNIX 最強大的特性之一)更難掌握。上面這樣的不一致問題甚至會給 UNIX 老手帶來困擾。
好在,fish (Friendly Interactive Shell) 的出現改變了這種混亂局面,它提供簡明的語法,顯著改善了用戶體驗。與其他 shell 一樣,fish 也提供重定向、快捷方式、globbing(即通配符的展開)、子 shell、製表符補全和變數。但是,與其他 shell 不同,fish 還提供顏色編碼的 CLI、功能豐富的命令行編輯器和大量文檔。
另外,對於執行任何操作,fish 只提供一種方式,這非常明智。如果一個 UNIX 實用程序能夠完成某一任務,fish 就不會通過內置命令重複提供此特性。例如,fish 使用系統範圍的應用程序 /bin/kill 終止進程。(與之相反,Bourne shell 通過一個內置應用程序實現了它自己的 kill 版本。可以在 Bourne shell 命令提示上輸入 /bin/kill 來訪問此版本)。fish 儘可能優先考慮簡單性而不是靈活性,這顯著簡化了它的使用方法。
下面,我們來安裝 fish 並體驗它的一些特性。
獲取 fish
fish 是由 Axel Liljencrantz 創建的一個開放源碼項目,採用的許可協議是 GNU General Public License, version 2。到編寫本文時,fish 的最新版本是 1.23.0,此版本於 2008 年 1 月 13 日發布。
如果使用 UNIX 或 UNIX 類系統(比如 Linux® 或 Mac OS X),那麼應該很容易在您的系統上從源代碼構建 fish。下面是構建步驟,見 清單 1:
bash-1) wget http://www.fishshell.org/files/1.23.0/fish-1.23.0.tar.gz bash-2) tar xzvf fish-1.23.0.tar.gz bash-3) cd fish-1.23.0 bash-4) ./configure --without-xsel checking if autoconf needs to be run... no checking if autoheader needs to be run... no checking for /usr/pkg/include include directory... no ... bash-5) make gcc -c -o function.o function.c ... bash-6) sudo make install ... To use fish as your login shell: * add the line '/usr/bin/fish' to the file '/etc/shells'. * use the command 'chsh -s /usr/bin/fish'. |
如果使用 UNIX 類系統,configure 可能不需要更多的標誌。但是,為了儘可能減少依賴性並讓 fish 採用與常用 shell 相同的目錄結構,可以分別添加 --without-xsel 和 --prefix=/usr。(如果使用 Mac OS X version 10.4 Leopard,那麼還要添加參數 LDFLAGS=-liconv。如果在 Mac OS X 上省略后一個選項,那麼無法編譯附帶的 fish 實用程序)。
另外,如果使用流行的 UNIX 版本,很可能會找到預先構建好的二進位版本,可以把它直接安裝在您的系統上。例如,如果使用 Debian Linux,那麼可以用 sudo apt-get install fish 命令直接安裝 fish。請訪問 fish 項目的主頁,了解是否有針對您的系統的二進位版本。
入門
在討論比較複雜的主題之前,我們先看看在 fish 中如何完成一些常見的 shell 任務:
在命令 3 中,rm 產生的錯誤消息被重定向到 errors 文件中。命令 4 顯示此文件的內容。fish shell 為重定向提供各種支持,比如把描述符組合成一個流和結束描述符。
順便說一句,文本的顏色和下劃線不是編輯出來的。shell 會在您輸入時在 CLI 中突出顯示文本。綠色表示命令名是有效的;無效的命令名用紅色表示。下劃線表示指定的文件存在。(後面一節詳細討論 shell 的反饋)。
函數可以包含一個或多個命令,特殊變數 $argv 會自動展開成命令行上傳遞的參數列表。
可以用 functions 命令列出已定義的所有函數。使用 functions --erase name 刪除函數,例如 functions --erase ll。
還可以立即保存在命令行上編寫的任何函數。在編寫完代碼時,輸入 funcsave name,例如 funcsave ll。在此之後,當前運行的所有 shell 和以後的所有 shell 都可以使用此函數。可以使用 funced name 命令以交互方式編輯現有的函數。funced 命令提供語法突出顯示、製表符補全和自動縮進;funcsave 和 funced 使用戶能夠更方便地定製 shell。
fish 提供 --query 選項來檢查是否定義了一個變數。如果已經設置了此變數,set --query 返回狀態碼 0,這表示沒有出現錯誤;否則,返回 1。語句 6 用 or 操作符連接兩個命令:第二個命令(echo)只在第一個命令失敗的情況下執行。
那麼,fish 如何處理 $var、'$var'、"$var" 和 `$var` 呢?它遵守幾條簡單的規則:
我們來看看這些規則的實際應用。
命令 1 創建四個 文件,最後一個文件的名稱包含空格。命令 3 和 4 刪除 file 變數指定的文件。命令 6 和 7 刪除 twofiles 變數指定的兩個文件。仔細看一下命令 6:因為值沒有放在引號(單引號或雙引號)中,所以不保留空格。因此,命令 7 把此變數展開成兩個參數並刪除兩個文件。命令 9 和 10 重複命令 6 和 7 中的場景。
命令 11 和 12 演示空格規則。儘管在命令 12 中變數沒有放在雙引號中,但是 fish 在命令 11 中保留空格。非常好。
命令 14 到 16 演示 fish 的嵌套引號規則。現在,再看一下命令 11、15 和 16。shell 使用顏色編碼顯示匹配的引號,以此確保語法正確。再看一下命令 9 和 11。后一個命令在文件名上顯示下劃線,這表示此文件存在。在命令 9 中沒有下劃線,這提示用戶某些地方出錯了。
fish 的首字母代表 Friendly,對用戶友好是它的主要目標。
對於新手非常有幫助的一個特性
說到對用戶友好,就不能不提到 fish 的製表符補全 特性,這個新穎的特性對於 UNIX 新用戶和專家都極其有幫助。為了體驗製表符補全,請按下面的示例操作。在每行的末尾按 Tab 鍵。
如果您不確定一個命令名的拼寫,可以在輸入幾個字母之後按 Tab,就會看到可能的完整命令的列表,見 圖 5。(在您的系統上顯示的命令列表可能與這裡顯示的不一樣。此列表取決於 PATH 環境變數和您的 UNIX 系統的內容)。
注意 CLI 中的紅色文本。如果 fish 不認識一個命令名,就用紅色顯示它。按 Tab,就會顯示以目前輸入的字母開頭的所有應用程序名(以及簡短的描述)。在空提示行上按 Tab,就會看到 PATH 中的所有應用程序。
如果想了解一個命令的可用選項,那麼在連字元(-)或雙連字元(--)後面按 Tab,見 圖 6。
此時,fish 會顯示可用的選項。shell 維護許多常用命令和選項的索引,您很可能能夠得到所需的幫助。但是,定製的或更複雜的實用程序可能缺少這種數據。可以閱讀 fish 文檔,了解關於編寫自己的補全特性的更多信息。
還可以在輸入選項的幾個字母之後按 Tab,見 圖 7。shell 會顯示所有匹配的選項。
如果您不知道一個命令處理的操作數類型,fish 在許多情況下可以提供幫助,但並不是在所有情況下都可以。例如,如果輸入 set(或 vared,即 fish 變數編輯器)和一個空格,然後按 Tab,fish 會顯示可用變數的列表。set 的操作數是一個變數。同樣,如果輸入 type 和一個空格,然後按 Tab,fish 會顯示內置函數的列表,這些函數擴展文件系統上可用的實用程序。
在一般情況下,fish 中的所有內置函數都有上下文相關的操作數補全。請試一下 cd,見 圖 8。
cd 函數是一個 fish 函數,它的操作數是一個現有的目錄。在輸入 cd 之後按 Tab,fish 會顯示 CDPATH 中的每個目錄包含的所有現有目錄。
另一個智能化補全與 ssh 相關。輸入 ssh 和一個空格,然後按 Tab,就會看到從 Secure Shell 已知主機文件(通常在 ~/.ssh/known_hosts 中)獲取的已知主機名列表:
fish-1) ssh login.example.com (Hostname) host1.example.com (Hostname) |
fish shell 還會補全文件名和目錄名。同樣,它會在您輸入路徑名時突出顯示正確的元素。
fish 與其他 shell 之間的一個重要差異是,它不提供歷史快捷方式,比如 !、!! 和 !$。
使用 fish 作為登錄 shell
如果您喜歡 fish,希望用它作為登錄 shell,那麼把 fish 的路徑添加到正式 shell 列表(/etc/shells)中,然後運行 chsh:
bash-1) type fish fish is /usr/bin/fish bash-2) sudo vi /etc/shells Add the line /usr/bin/fish to the file if it's missing, and save the file bash-3) cat /etc/shells /bin/bash /bin/csh /bin/ksh /bin/sh /bin/tcsh /bin/zsh /usr/bin/fish bash-4) chsh -s /usr/bin/fish Changing shell for strike Password: ******** bash-5) login strike Password: ******** Last login: Wed Oct 8 15:02:21 on ttys000 Welcome to fish, the friendly interactive shell Type help for instructions on how to use fish fish-1) echo $SHELL /usr/bin/fish |
結束語
fish 中還有許多有用的特性值得研究。“fish 這條魚非常有營養。”
可以調整語法突出顯示採用的顏色。可以通過編輯 ~/.config/fish/config.fish 定製啟動過程。可以使用通用變數 和 fishd 跨 shell 實例共享變數。這種 shell 還有出色的歷史搜索特性、互動式變數編輯器和互動式命令行編輯器。
最好的一點是,fish 本身提供大量文檔。如果需要幫助,只需在命令提示上輸入 help。
醫生的意見是對的:吃 “魚” 對您有益。(責任編輯:A6)
[火星人 ] 對話 UNIX: 使用 fish!已經有756次圍觀