為什麼應該在 Linux 上使用命名管道

Sandra Henry-stocker發表於2018-02-05

命名管道並不常用,但是它們為程式間通訊提供了一些有趣的特性。

估計每一位 Linux 使用者都熟悉使用 “|” 符號將資料從一個程式傳輸到另一個程式的操作。它使使用者能簡便地從一個命令輸出資料到另一個命令,並篩選出想要的資料而無須寫指令碼進行選擇、重新格式化等操作。

還有另一種管道, 雖然也叫“管道”這個名字卻有著非常不同的性質。即您可能尚未使用甚至尚未知曉的——命名管道。

普通管道與命名管道的一個主要區別就是命名管道是以檔案形式實實在在地存在於檔案系統中的,沒錯,它們表現出來就是檔案。但是與其它檔案不同的是,命名管道檔案似乎從來沒有檔案內容。即使使用者往命名管道中寫入大量資料,該檔案看起來還是空的。

如何在 Linux 上建立命名管道

在我們研究這些空空如也的命名管道之前,先追根溯源來看看命名管道是如何被建立的。您應該使用名為 mkfifo 的命令來建立它們。為什麼提及“FIFO”?是因為命名管道也被認為是一種 FIFO 特殊檔案。術語 “FIFO” 指的是它的先進先出first-in, first-out特性。如果你將冰淇淋盛放到碟子中,然後可以品嚐它,那麼你執行的就是一個LIFO(後進先出last-in, first-out操作。如果你透過吸管喝奶昔,那你就在執行一個 FIFO 操作。好,接下來是一個建立命名管道的例子。

$ mkfifo mypipe
$ ls -l mypipe
prw-r-----. 1 shs staff 0 Jan 31 13:59 mypipe

注意一下特殊的檔案型別標記 “p” 以及該檔案大小為 0。您可以將重定向資料寫入命名管道檔案,而檔案大小依然為 0。

$ echo "Can you read this?" > mypipe

正如上面所說,敲擊回車後似乎什麼都沒有發生(LCTT 譯註:沒有返回命令列提示符)。

另外再開一個終端,檢視該命名管道的大小,依舊是 0:

$ ls -l mypipe
prw-r-----. 1 shs staff 0 Jan 31 13:59 mypipe

也許這有違直覺,使用者輸入的文字已經進入該命名管道,而你仍然卡在輸入端。你或者其他人應該等在輸出端,並準備讀取放入管道的資料。現在讓我們讀取看看。

$ cat mypipe
Can you read this?

一旦被讀取之後,管道中的內容就沒有了。

另一種研究命名管道如何工作的方式是透過將放入資料的操作置入後臺來執行兩個操作(將資料放入管道,而在另外一段讀取它)。

$ echo "Can you read this?" > mypipe &
[1] 79302
$ cat mypipe
Can you read this?
[1]+ Done echo "Can you read this?" > mypipe

一旦管道被讀取或“耗幹”,該管道就清空了,儘管我們還能看見它並再次使用。可為什麼要費此周折呢?

為何要使用命名管道?

命名管道很少被使用的理由似乎很充分。畢竟在 Unix 系統上,總有多種不同的方式完成同樣的操作。有多種方式寫檔案、讀檔案、清空檔案,儘管命名管道比它們來得更高效。

值得注意的是,命名管道的內容駐留在記憶體中而不是被寫到硬碟上。資料內容只有在輸入輸出端都開啟時才會傳送。使用者可以在管道的輸出端開啟之前向管道多次寫入。透過使用命名管道,使用者可以建立一個程式寫入管道並且另外一個程式讀取管道的流程,而不用關心協調二者時間上的同步。

使用者可以建立一個單純等待資料出現在管道輸出端的程式,並在拿到輸出資料後對其進行操作。下列命令我們採用 tail 來等待資料出現。

$ tail -f mypipe

一旦供給管道資料的程式結束了,我們就可以看到一些輸出。

$ tail -f mypipe
Uranus replicated to WCDC7
Saturn replicated to WCDC8
Pluto replicated to WCDC9
Server replication operation completed

如果研究一下向命名管道寫入的程式,使用者也許會驚訝於它的資源消耗之少。在下面的 ps 命令輸出中,唯一顯著的資源消耗是虛擬記憶體(VSZ 那一列)。

ps u -P 80038
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
shs 80038 0.0 0.0 108488 764 pts/4 S 15:25 0:00 -bash

命名管道與 Unix/Linux 系統上更常用的管道相比足以不同到擁有另一個名號,但是“管道”確實能反映出它們如何在程式間傳送資料的形象,故將稱其為“命名管道”還真是恰如其分。也許您在執行操作時就能從這個聰明的 Unix/Linux 特性中獲益匪淺呢。


via: https://www.networkworld.com/article/3251853/linux/why-use-named-pipes-on-linux.html

作者:Sandra Henry-Stocker 譯者:YPBlib 校對:wxy

本文由 LCTT 原創編譯,Linux中國 榮譽推出

相關文章