那些奇奇怪怪的緩沖問(wèn)題
在上一篇《不可不知的緩沖類型》中說(shuō)到了三種緩沖類型,看起來(lái)很簡(jiǎn)單對(duì)不對(duì)?今天我們看看如何修改這些默認(rèn)的緩沖類型,以及在實(shí)際中可能遇到哪些問(wèn)題。
更改緩沖類型
在上一篇中說(shuō)到了一些默認(rèn)的緩沖類型,例如:
- 指向終端設(shè)備的流是行緩沖的
- 標(biāo)準(zhǔn)錯(cuò)誤是不帶緩沖的
- 指向文件的流是全緩沖的
- ……
那么這些默認(rèn)的緩沖類型如何修改?有幾個(gè)函數(shù)可以用來(lái)更改緩沖類型:
- #include<stdio.h>
- void setbuf(FILE *stream, char *buf);
- void setbuffer(FILE *stream, char *buf, size_t size);
- void setlinebuf(FILE *stream);
- int setvbuf(FILE *stream,char *buf, int mode, size_t size);
參數(shù)說(shuō)明如下:
- stream FILE *類型,文件指針
- buf 緩沖區(qū)指針
- mode 緩沖模式,包括_IOFBF(全緩沖),_IOLBF(行緩沖),_IONBF(不帶緩沖)
- size 緩沖區(qū)大小
這里有四個(gè)相關(guān)函數(shù),作用類似,只是修改范圍不一。
setbuf函數(shù)中,如果buf設(shè)置為NULL,則緩沖關(guān)閉;否則指向長(zhǎng)度為BUFSIZ長(zhǎng)度的緩沖區(qū),并且是行緩沖。
- #include<stdio.h>
- #include<unistd.h>
- int main(void)
- {
- setbuf(stdout,NULL);
- printf("bianchengzhuji");
- sleep(10);
- return 0;
- }
這里設(shè)置為不帶緩沖,則會(huì)忽略buf和size參數(shù)。設(shè)置為全緩沖或者行緩沖的時(shí)候。并且buf為NULL,會(huì)使用合適長(zhǎng)度的系統(tǒng)buffer,否則使用用戶自定義buffer。緩沖區(qū)的設(shè)置就介紹到這里。
fputs沒(méi)有及時(shí)輸出
其實(shí)在有了前面的基礎(chǔ)之后,很多問(wèn)題就迎刃而解了。
看看下面的例子:
- #include<stdio.h>
- #include<unistd.h>
- int main(void)
- {
- //setbuf(stdout,NULL);
- fputc('a',stdout);
- sleep(10);
- return 0;
- }
比如你就想輸出一個(gè)字符,就打印到終端,但是按照上面的方法,字符并不會(huì)被及時(shí)輸出到終端,因此它默認(rèn)是行緩沖的。打開(kāi)注釋行,設(shè)置為不帶緩沖就可以了。
printf打印的日志沒(méi)有輸出
不知道你有沒(méi)有遇到過(guò)這樣的情況,準(zhǔn)備調(diào)試某一個(gè)bug,發(fā)現(xiàn)每次運(yùn)行到某個(gè)地方,打印就結(jié)束了,然后就掛了,讓你誤以為程序執(zhí)行到打印的地方就結(jié)束了,然而有可能程序執(zhí)行到后面,只是由于打印是行緩沖的,導(dǎo)致部分打印沒(méi)有出來(lái),很可能就是你沒(méi)有加上換行符打印而已。
這時(shí)候你可以設(shè)置為不帶緩沖,或者關(guān)鍵位置fflush,或者打印記得加上換行符。
fflush之后文件還是丟失了
看完前面的內(nèi)容之后,是不是覺(jué)得豁然開(kāi)朗了?別高興的太早。
以上措施并不是萬(wàn)事大吉。
你可能會(huì)踩到什么坑?
- 文件內(nèi)容寫(xiě)完后,fflush了,內(nèi)容也有了,然后完成后,系統(tǒng)馬上復(fù)位,復(fù)位起來(lái)后,文件內(nèi)容還是丟失了
- 解壓一個(gè)壓縮包,解壓成功,系統(tǒng)復(fù)位后,還是發(fā)現(xiàn)文件大小為0,文件丟失了
如果你目前還沒(méi)有遇到過(guò)這樣的問(wèn)題,那么你就需要格外注意了。雖然前面fflush等措施將緩沖區(qū)的內(nèi)容進(jìn)行了I/O操作,但是操作系統(tǒng)還需要將文件系統(tǒng)的buffer寫(xiě)入磁盤,因此馬上直接復(fù)位會(huì)導(dǎo)致文件丟失!怎么辦呢?可以使用:
- fsync/sync函數(shù)
- sync命令
以上函數(shù)或者命令強(qiáng)制將文件系統(tǒng)的buffer寫(xiě)入磁盤,但是根據(jù)內(nèi)容大小不一而需要不一樣的時(shí)間。