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

關(guān)于流和緩沖區(qū)的理解

0. 序曲

寫這篇短文的起因是,前兩天想去天大的acm在線系統(tǒng)找?guī)椎李}做做。為什么呢?因為本人天大畢業(yè),這個天大呢可是中國最早的大學(xué),原名北洋大學(xué)堂,這可絕對是貨真價實的第一所大學(xué)。給大家推薦推薦啊,學(xué)風(fēng)那是相當(dāng)?shù)暮谩?/p>

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


 

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

 


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

輸出gfik。

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


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


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


聲明:

本文很大部分內(nèi)容來自APUE--UNIX環(huán)境高級編程。

1. 緩沖類型。

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

  • 全緩沖。

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

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

  • 行緩沖

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

  • 無緩沖

標準I/O庫不緩存字符。需要注意的是,標準庫不緩存并不意味著操作系統(tǒng)或者設(shè)備驅(qū)動不緩存。

ISO C要求:

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

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

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

2. 改變默認緩存類型

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

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

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

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

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

關(guān)于這兩個函數(shù),可以看下表(摘自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

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


3.如果清理輸入緩存

關(guān)于這點可以參看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操作是無定義的。
  • 無緩存并不意味著一個個的那樣處理輸入,而是說當(dāng)操作系統(tǒng)返回它們時,對于標準庫函數(shù)來說它們是立即可用的。因為還可能有操作系統(tǒng)級甚至是硬件級的緩存,這些并不是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 季陽 閱讀(6418) 評論(4)  編輯 收藏 引用

評論

# re: 關(guān)于流和緩沖區(qū)的理解 2008-04-07 15:03 亨德列克

受教了  回復(fù)  更多評論   

# re: 關(guān)于流和緩沖區(qū)的理解 2008-04-07 15:51 yyxxh

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

# re: 關(guān)于流和緩沖區(qū)的理解 2011-10-11 15:12 靜靜

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

# re: 關(guān)于流和緩沖區(qū)的理解 2011-12-26 19:48 金艷

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


只有注冊用戶登錄后才能發(fā)表評論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


<2008年4月>
303112345
6789101112
13141516171819
20212223242526
27282930123
45678910

導(dǎo)航

統(tǒng)計

常用鏈接

留言簿(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>
            亚洲第一久久影院| 亚洲一区二区成人在线观看| 欧美黄污视频| 最新国产成人av网站网址麻豆| 亚洲激情成人网| 亚洲午夜电影| 久久深夜福利| 亚洲精选在线观看| 日韩亚洲在线| 欧美日本精品一区二区三区| 亚洲三级电影在线观看| 欧美阿v一级看视频| 欧美一区二区在线观看| 久久人人爽人人| 美女主播一区| 欧美一区二区三区四区视频| 久久久国产91| 欧美一区二区高清在线观看| 麻豆国产精品777777在线| 亚洲日产国产精品| 国产亚洲一区在线| 9人人澡人人爽人人精品| 亚洲第一级黄色片| 欧美在线日韩精品| 欧美一区二区高清| 国产精品成人在线观看| 欧美国产一区视频在线观看| 国产伊人精品| 亚洲欧美在线磁力| 亚洲一区二区三区中文字幕在线 | 欧美成人黑人xx视频免费观看| 免费观看久久久4p| 国产精品一区久久| 亚洲视频电影在线| 性亚洲最疯狂xxxx高清| 国产精品免费小视频| 亚洲欧美日本国产有色| 亚洲影院免费观看| 国产精品爽爽爽| 午夜精品久久久久久久久久久久| 亚洲一区二区三区四区中文| 欧美日韩一区在线播放| 亚洲视频一二三| 久久aⅴ国产欧美74aaa| 国产乱码精品| 欧美成人中文| 亚洲影视九九影院在线观看| 欧美在线高清| 亚洲国产成人精品久久| 久久国内精品自在自线400部| 国产精品丝袜xxxxxxx| 亚洲一区免费网站| 久久久久天天天天| 亚洲日韩视频| 国产精品初高中精品久久| 午夜精品久久久久久久白皮肤 | 99伊人成综合| 国产丝袜一区二区| 欧美三级第一页| 久久久精品999| 亚洲欧美第一页| 欧美成人免费在线观看| 亚洲午夜国产成人av电影男同| 韩国v欧美v日本v亚洲v| 国产精品高清网站| 欧美日韩在线播放三区| 老司机免费视频一区二区| 亚洲一区精品视频| 亚洲二区三区四区| 美女精品自拍一二三四| 久久日韩粉嫩一区二区三区| 久久精品成人一区二区三区| 亚洲一区三区视频在线观看| 亚洲最新视频在线| 亚洲一二三区在线| 亚洲一区二区动漫| 久久伊人免费视频| 欧美一区三区二区在线观看| 亚洲永久免费观看| 亚洲在线观看视频| 亚洲一级二级在线| 欧美一区二区私人影院日本| 亚洲一区二区三区免费观看 | 国产午夜精品全部视频在线播放| 日韩午夜视频在线观看| 免费日本视频一区| 亚洲日韩成人| 久久蜜桃香蕉精品一区二区三区| 免费看精品久久片| 欧美另类在线播放| 精品福利电影| 亚洲资源av| 亚洲精品乱码久久久久久黑人| 一二三区精品福利视频| 欧美在线视频免费| 欧美二区乱c少妇| 国产精品久久一级| 日韩亚洲在线| 久久九九久久九九| 亚洲一区在线视频| 免费在线播放第一区高清av| 欧美午夜激情在线| 在线观看视频欧美| 亚洲欧美春色| aⅴ色国产欧美| 欧美日韩视频第一区| 亚洲乱码日产精品bd| 欧美成人69av| 久久免费视频一区| 一色屋精品视频在线看| 久久久久国色av免费看影院| 99精品国产99久久久久久福利| 久久久久久**毛片大全| 国产毛片久久| 午夜视频一区在线观看| 亚洲神马久久| 国产精品二区影院| 午夜在线一区| 久久成人在线| 黄色欧美日韩| 久久免费视频一区| 欧美成人亚洲| 久久狠狠亚洲综合| 欧美sm极限捆绑bd| 99热在线精品观看| 亚洲欧美不卡| 亚洲九九精品| 久久精品视频一| 欧美日韩福利在线观看| 亚洲一区二区三区久久| 亚洲欧美日韩视频一区| 在线播放一区| 亚洲福利视频一区| 国产午夜亚洲精品不卡| 欧美成人午夜77777| 国产精品自在欧美一区| 亚洲国产女人aaa毛片在线| 国产精品v一区二区三区| 久久久www成人免费无遮挡大片| 毛片一区二区三区| 久久夜色精品国产欧美乱极品| 欧美精品三级日韩久久| 欧美夫妇交换俱乐部在线观看| 欧美视频在线看| 欧美大片一区二区三区| 国产精品私拍pans大尺度在线 | 久久一区二区三区超碰国产精品| 欧美高清不卡| 免费亚洲电影在线| 国产综合色产在线精品| 亚洲在线观看视频| 久久大逼视频| 国产一区二区三区视频在线观看| 日韩一区二区福利| 亚洲高清免费在线| 欧美va亚洲va香蕉在线| 久久大逼视频| 国模私拍视频一区| 久久久久91| 亚洲黄色一区二区三区| 亚洲国产欧美精品| 欧美激情1区2区| 亚洲免费在线看| 亚洲国产精品www| 亚洲一区二区三区四区五区黄| 欧美性猛交xxxx免费看久久久| 亚洲欧美日韩综合国产aⅴ| 久久爱另类一区二区小说| 在线高清一区| 欧美午夜一区二区三区免费大片| 亚洲一区高清| 亚洲国产成人在线播放| 亚洲欧美精品伊人久久| 黄色成人免费观看| 久久久视频精品| 中日韩在线视频| 亚洲美女视频在线观看| 国产精品亚洲网站| 国产精品九九| 欧美理论大片| 麻豆精品一区二区综合av| 一区二区三区高清在线| 老鸭窝毛片一区二区三区| 亚洲无吗在线| 亚洲精品一区二区在线观看| 国内精品久久久久影院色| 欧美国产视频在线| 午夜欧美精品久久久久久久| 亚洲国产高清在线| 欧美大片在线影院| 欧美二区在线观看| 欧美国产日韩一区二区在线观看| 久久超碰97中文字幕| 欧美一级淫片aaaaaaa视频| 亚洲女同同性videoxma| 亚洲欧美国产制服动漫| 欧美一区二区三区四区在线| 欧美综合激情网| 免费成人毛片| av成人免费在线|