青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

關于流和緩沖區的理解

0. 序曲

寫這篇短文的起因是,前兩天想去天大的acm在線系統找幾道題做做。為什么呢?因為本人天大畢業,這個天大呢可是中國最早的大學,原名北洋大學堂,這可絕對是貨真價實的第一所大學。給大家推薦推薦啊,學風那是相當的好。

扯多了,還是回到本來的話題上。上了acm系統之后,就先看了1001。那道題的意思是輸入一些正整數(以EOF結束),把對應的字符輸出。這個簡單,程序很快就出來了:


 

#include <stdio.h>
int main()
{
int c;
while(scanf("%d"&c) != EOF)
{
putchar(c);
}
return 0;
}

 


程序運行,輸入103 102 105 107<enter>

輸出gfik。

當時運行完之后馬上想,為什么不是輸入一個數字馬上輸出一個字符呢,因為看程序確實是這樣的邏輯,只要不是EOF,就會輸出。又一想,對了,是緩沖的問題。想起來APUE里邊說得stdin應該是行緩沖的,另外,可以用setbuf,setvbuf設定流的緩沖。于是想將stdin設成無緩沖的。于是程序變成這樣:


#include <stdio.h>
int main()
{
int c;
setbuf(stdin, NULL);
while(scanf("%d"&c) != EOF)
{
putchar(c);
}
return 0;
}


可是編譯運行,還是老樣子,沒有變化。想了想,沒想出是啥原因,于是開始google和APUE。終于算是明白了些,整理在這兒。


聲明:

本文很大部分內容來自APUE--UNIX環境高級編程。

1. 緩沖類型。

標準庫提供緩沖是為了減少對read和write的調用。提供的緩沖有三種類型(整理自APUE):

  • 全緩沖。

在這種情況下,實際的I/O操作只有在緩沖區被填滿了之后才會進行。對駐留在磁盤上的文件的操作一般是有標準I/O庫提供全緩沖。緩沖區一般是在第一次對流進行I/O操作時,由標準I/O函數調用malloc函數分配得到的。

術語flush描述了標準I/O緩沖的寫操作。緩沖區可以由標準I/O函數自動flush(例如緩沖區滿的時候);或者我們對流調用fflush函數。

  • 行緩沖

在這種情況下,只有在輸入/輸出中遇到換行符的時候,才會執行實際的I/O操作。這允許我們一次寫一個字符,但是只有在寫完一行之后才做I/O操作。一般的,涉及到終端的流--例如標注輸入(stdin)和標準輸出(stdout)--是行緩沖的。

  • 無緩沖

標準I/O庫不緩存字符。需要注意的是,標準庫不緩存并不意味著操作系統或者設備驅動不緩存。

ISO C要求:

  • 當且僅當不涉及交互設備時,標準輸入和標準輸出是全緩存的。
  • 標準錯誤絕對不是全緩存的。

但是,這并沒有告訴我們當標準輸入/輸出在涉及交互設備時,它們是無緩存的還是行緩存的;也沒有告訴我們標準錯誤應該是行緩存的還是無緩存的。不過,大多數實現默認的緩存類型是這樣的:

  • 標準錯誤總是無緩存的。
  • 對于所有的其他流來說,如果它們涉及到交互設備,那么就是行緩存的;否則是全緩存的。

2. 改變默認緩存類型

可以通過下面的函數改變緩存類型(摘自APUE):

void setbuf(FILE *restrict fp, char *restrict buf);
int setvbuf(FILE *restrict fp, char *restrict buf, int mode, size_t size);

這些函數必須在流打開之后、但是未對流做任何操作之前被調用(因為每個函數都需要一個有效的文件指針作為第一個參數)。

利用setbuf,可以打開或者關閉緩存。為了打開緩存,buf參數必須一個大小為BUFSIZ的緩存,BUFSIZ是定義在stdio。h中的常量。&amp;lt;&amp;lt;ISO/IEC 9899&amp;gt;&amp;gt;要求:BUFSIZ至少為256。如果要關閉緩存,可以將buf設成NULL。

利用setvbuf,我們可以設定緩存類型。這是通過mode參數指定的。

關于這兩個函數,可以看下表(摘自APUE):


Function

mode

buf

Buffer and length

Type of buffering

setbuf

non-null

user buf of length BUFSIZ

fully buffered or line buffered

NULL

(no buffer)

unbuffered

setvbuf

_IOLBF

non-null

user buf of length size

fully buffered

NULL

system buffer of appropriate length

_IOFBF

non-null

user buf of length size

line buffered

NULL

system buffer of appropriate length

_IONBF

(ignored)

(no buffer)

unbuffered

需要注意的是:如果在函數內為流分配了自動變量作為緩存,那么在退出之前需要將流關閉。因此最好讓系統自己分配緩存,這些緩存在流關閉的時候會自動被釋放。


3.如果清理輸入緩存

關于這點可以參看comp.lang.c FAQ的Question12.26b:

Q: If fflush won't work, what can I use to flush input?

A: It depends on what you're trying to do. If you're trying to get rid of an unread newline or other unexpected input after calling scanf (see questions 12.18a-12.19), you really need to rewrite or replace the call to scanf (see question 12.20). Alternatively, you can consume the rest of a partially-read line with a simple code fragment like

while((c = getchar()) != '\n' &amp;&amp; c != EOF)
/* discard */ ;

(You may also be able to use the curses flushinp function.)

There is no standard way to discard unread characters from a stdio input stream. Some vendors do implement fflush so that fflush(stdin) discards unread characters, although portable programs cannot depend on this. (Some versions of the stdio library implement fpurge or fabort calls which do the same thing, but these aren't standard, either.) Note, too, that flushing stdio input buffers is not necessarily sufficient: unread characters can also accumulate in other, OS-level input buffers. If you're trying to actively discard input (perhaps in anticipation of issuing an unexpected prompt to confirm a destructive action, for which an accidentally-typed ``y'' could be disastrous), you'll have to use a system-specific technique to detect the presence of typed-ahead input; see questions 19.1 and 19.2. Keep in mind that users can become frustrated if you discard input that happened to be typed too quickly.

References: ISO Sec. 7.9.5.2
H&amp;S Sec. 15.2

4. 幾點需要注意的地方

  • 對輸入流進行fflush操作是無定義的。
  • 無緩存并不意味著一個個的那樣處理輸入,而是說當操作系統返回它們時,對于標準庫函數來說它們是立即可用的。因為還可能有操作系統級甚至是硬件級的緩存,這些并不是setbuf可以控制的。
  • 另外可以參考這里(我就是最先從這里開始看的)。還有這里。我從后面那個鏈接摘錄一些重要的下來:

setbuf() has to do with the delivery of bytes between the
C library FILE* management layer and the OS I/O layer.

Calls to fread(), fgets(), fgetc(), and getchar() work within
whatever FILE* buffered data is available, and when that data
is exhausted, the calls request that the FILE* buffer be refilled
by the system I/O layer.

When full buffering is turned on, that refill operation results in the
FILE* layer requesting that the operating system hand it a full
buffer's worth of data; when buffering is turned off, that
refill operation results in the FILE* layer requesting that the
operating system return a single character.

...setting an input stream to be unbuffered
does NOT tell the operating system to tell the device driver
to go into any kind of "raw" single-character mode. There are
system-specific calls such as ioctl() and tcsetterm() that
control what the device driver will do.





posted on 2008-04-07 12:48 季陽 閱讀(6416) 評論(4)  編輯 收藏 引用

評論

# re: 關于流和緩沖區的理解 2008-04-07 15:03 亨德列克

受教了  回復  更多評論   

# re: 關于流和緩沖區的理解 2008-04-07 15:51 yyxxh

上次看譚浩強的C程序設計例子也提了這樣一個問題,再次看到有人提出這樣的問題,溫故而知新啊  回復  更多評論   

# re: 關于流和緩沖區的理解 2011-10-11 15:12 靜靜

int main()
{
int c;
setbuf(stdin, NULL);
while(scanf("%d", &c) != EOF)
{
putchar(c);
}
return 0;
}
為啥輸出還是一樣呢?setbuf(stdin,NULL)沒發揮作用嗎?樓主解釋一下。  回復  更多評論   

# re: 關于流和緩沖區的理解 2011-12-26 19:48 金艷

對啊,為什么第二個例子不起作用還沒說啊  回復  更多評論   

<2011年10月>
2526272829301
2345678
9101112131415
16171819202122
23242526272829
303112345

導航

統計

常用鏈接

留言簿(2)

隨筆檔案(12)

搜索

最新隨筆

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            很黄很黄激情成人| 国产精品综合| 99人久久精品视频最新地址| 欧美激情偷拍| 欧美精品三区| 午夜日韩激情| 久久久久国色av免费观看性色| 极品av少妇一区二区| 欧美顶级少妇做爰| 欧美日韩一区二区精品| 午夜久久美女| 久久久国产精品一区二区三区| 亚洲国产精品尤物yw在线观看| 亚洲国产天堂网精品网站| 久久久噜噜噜久久久| 久久精品99久久香蕉国产色戒| 亚洲国产高清在线观看视频| 亚洲老司机av| 黑人一区二区三区四区五区| 亚洲国内自拍| 国产一区二区三区成人欧美日韩在线观看 | 一本久久a久久免费精品不卡| 国产精品中文字幕欧美| 蜜桃精品一区二区三区| 欧美日韩在线视频一区二区| 久久久久久久波多野高潮日日| 欧美福利视频一区| 欧美制服第一页| 欧美国产一区二区在线观看 | 中文国产一区| 久久先锋影音av| 欧美一区91| 欧美日韩的一区二区| 久久久久久久999| 国产精品高潮粉嫩av| 欧美激情在线有限公司| 国产亚洲欧美日韩美女| 亚洲视频电影图片偷拍一区| 亚洲高清影视| 久久精品在线视频| 欧美一区二区三区精品电影| 欧美伦理a级免费电影| 蜜桃av一区二区| 国产色视频一区| 在线亚洲免费| 一区二区免费在线观看| 欧美v亚洲v综合ⅴ国产v| 麻豆91精品91久久久的内涵| 国产精品久久久久久久一区探花| 亚洲黄色影片| 亚洲精品专区| 欧美成人激情视频| 蜜桃精品久久久久久久免费影院| 国产欧美日韩中文字幕在线| 亚洲天堂第二页| 亚洲私人影院| 欧美亚男人的天堂| 99热免费精品| 亚洲专区欧美专区| 国产精品黄色| 亚洲午夜在线视频| 性欧美在线看片a免费观看| 欧美亚洲成人网| 中文精品99久久国产香蕉| 一区二区三区.www| 国产精品国产三级欧美二区| 一区二区三区精品| 亚洲欧洲av一区二区| 国产精品久久久久久久久久久久 | 亚洲第一级黄色片| 亚洲精品欧洲| 欧美三级在线播放| 亚洲一区二区三区免费观看| 香蕉乱码成人久久天堂爱免费| 国产精品五区| 亚洲福利国产| av成人动漫| 国产精品成人一区| 欧美一区激情视频在线观看| 乱码第一页成人| 亚洲精品一区二区三区蜜桃久| 欧美精品www在线观看| 一区二区三区日韩| 久久久91精品| 91久久精品日日躁夜夜躁国产| 欧美久久一级| 欧美一区二区三区四区夜夜大片| 麻豆成人小视频| 99精品久久免费看蜜臀剧情介绍| 国产精品久久国产精麻豆99网站| 午夜精品久久久久久99热| 女生裸体视频一区二区三区| 亚洲精品视频免费在线观看| 国产精品第一区| 久久综合中文字幕| 一本色道久久综合亚洲精品高清| 欧美一区二区三区在线免费观看| 在线精品一区二区| 国产精品国产三级国产普通话99| 久久国产福利| 日韩一区二区精品视频| 久久另类ts人妖一区二区 | 国产农村妇女毛片精品久久莱园子| 久久久精品网| 亚洲少妇诱惑| 亚洲福利视频专区| 欧美一区2区三区4区公司二百| 亚洲黄色影院| 狠狠色狠狠色综合系列| 国产精品国产三级国产普通话蜜臀 | 母乳一区在线观看| 亚洲综合色婷婷| 亚洲精品在线观看免费| 欧美jizzhd精品欧美巨大免费| 亚洲男女毛片无遮挡| 亚洲免费播放| 1204国产成人精品视频| 国产精品一区二区久久| 欧美日韩久久精品| 欧美高清在线一区| 久久久蜜桃精品| 欧美一区1区三区3区公司| 99在线精品免费视频九九视| 欧美电影打屁股sp| 久久综合一区| 久久躁日日躁aaaaxxxx| 欧美在线播放一区| 亚洲欧美日韩精品久久奇米色影视 | 国产精品久久国产三级国电话系列| 蜜桃久久精品一区二区| 久久蜜桃精品| 久久久精品日韩| 久久久夜色精品亚洲| 久久成人精品| 久久久www成人免费无遮挡大片| 亚洲午夜伦理| 亚洲一区综合| 先锋影音久久| 欧美伊人久久久久久午夜久久久久 | 久久国产精品一区二区三区四区| 亚洲综合视频一区| 亚洲在线免费观看| 亚洲欧美一区二区三区极速播放| 亚洲视频免费| 亚洲在线中文字幕| 午夜精品在线观看| 欧美影院成人| 久久天堂成人| 欧美黑人在线播放| 欧美日韩一区二区高清| 国产精品久久久久aaaa九色| 国产精品色一区二区三区| 国产免费观看久久黄| 韩国三级在线一区| 亚洲第一成人在线| 99视频+国产日韩欧美| 一本大道久久精品懂色aⅴ| 亚洲一区激情| 久久久久久久久久久久久女国产乱 | 亚洲欧洲在线看| 一区二区三区欧美日韩| 亚洲欧美国产日韩中文字幕| 久久超碰97人人做人人爱| 免费在线观看精品| 欧美三级欧美一级| 国外成人在线视频| 亚洲精品国产精品久久清纯直播 | 久久久亚洲综合| 亚洲国产日韩欧美| 亚洲午夜一区| 免费亚洲电影| 国产日产亚洲精品系列| 亚洲国产精品一区二区尤物区| 一区二区三区导航| 久久久噜噜噜久久狠狠50岁| 亚洲国产精品精华液2区45| 亚洲少妇诱惑| 嫩草国产精品入口| 国产欧美一区二区三区在线看蜜臀| 激情校园亚洲| 午夜免费久久久久| 亚洲国产高清aⅴ视频| 午夜老司机精品| 欧美日本亚洲视频| 激情文学综合丁香| 亚洲男人的天堂在线| 亚洲成色精品| 欧美大片免费| 亚洲欧美第一页| 欧美日本中文| 亚洲第一网站| 久久五月天婷婷| 亚洲男女自偷自拍| 欧美日韩一区国产| 亚洲日本在线观看| 久久久久久夜精品精品免费| 在线视频欧美精品| 欧美日本在线| 亚洲精品欧美精品| 欧美激情 亚洲a∨综合|