stdout, stdin, stderr的中文名字分別是標準輸出,標準輸入和標準錯誤。
在Linux下,當一個用戶進程被創建的時候,系統會自動爲該進程創建三個數據流,也就是題目中所提到的這三個。那麼什麼是數據流呢(stream)?我們知道,一個程序要運行,需要有輸入、輸出,如果出錯,還要能表現出自身的錯誤。這是就要從某個地方讀入數據、將數據輸出到某個地方,這就夠成了數據流。
因此,一個進程初期所擁有的這麼三個數據流,就分別是標準輸出、標準輸入和標準錯誤,分別用stdout, stdin, stderr來表示。對於這三個數據流來說,默認是表現在用戶終端上的,比如我們在c中使用fprintf:
fprintf(stdout,"hello world!\n");
屏幕上將打印出"hello world!"來,同樣,我們使用:
上面的代碼會接收用戶輸入在終端裏的字符,並存在ptr中。
fread(ptr,1,10,stdin);
那麼標準輸入輸出和錯誤是不是隻能反應在終端裏呢?答案是不是的!我們可以將標準輸入和輸出重定位到文件中:
例如,我們使用ls命令,會把當前目錄下的文件名輸出到終端上:
$ls
gccgcc.sh gmp-5.0.1 gmp-5.0.1.tar.bz2 linux-loongson-community-2.6.35-rc1 longene-0.3.0-linux-2.6.34 mpfr-3.0.0 mpfr-3.0.0.tar.gz oprofile-0.9.6
我們可以使用 “ > ”符號,將ls的標準輸出重定向到文件中:
$ls>lsout //將標準輸出重定向爲文件lsout
$morelsout //顯示lsout文件裏的內容
gccgcc.sh gmp-5.0.1 gmp-5.0.1.tar.bz2 linux-loongson-community-2.6.35-rc1 longene-0.3.0-linux-2.6.34 mpfr-3.0.0 mpfr-3.0.0.tar.gz oprofile-0.9.6
同樣,我們也可以使用“ < ”符號將標準輸入重定向到文件中,以sort爲例,以下示例使用 sort 命令對由鍵盤鍵入的文本進行排序。鍵入ctrl-D 結束標準輸入。終端屏幕顯示的標準輸出如下:
$sort
muffy
happy
bumpy
CTRL-D //結束標準輸入。
bumpy
happy
muffy //結束標準輸出。
使用" < "重定向後爲:
$ moresocks 顯示 socks 的內容。
polka dot
argyle
plaid
$ sort<socks 將輸入重定向爲從 socks 輸入,並將內容排序。
argyle
plaid
polka dot
好,基本知識講完了,我們知道,標準輸出和標準錯誤默認都是將信息輸出到終端上,那麼他們有什麼區別呢?讓我們來看個題目:
問題:下面程序的輸出是什麼?(intel筆試2011)
intmain(){
fprintf(stdout,"Hello ");
fprintf(stderr,"World!");
return0;
}
解答:這段代碼的輸出是什麼呢?你可以快速的將代碼敲入你電腦上(當然,拷貝更快),然後發現輸出是
World!Hello
這是爲什麼呢?在默認情況下,stdout是行緩衝的,他的輸出會放在一個buffer裏面,只有到換行的時候,纔會輸出到屏幕。而stderr是無緩衝的,會直接輸出,舉例來說就是printf(stdout, "xxxx") 和 printf(stdout, "xxxx\n"),前者會憋住,直到遇到新行纔會一起輸出。而printf(stderr, "xxxxx"),不管有麼有\n,都輸出。