那些奇奇怪怪的緩沖問(wèn)題
今天我們看看如何修改這些默認(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(全緩沖),_IOLBUF(行緩沖),_IONBF(不帶緩沖)
- size 緩沖區(qū)大小
這里有四個(gè)相關(guān)函數(shù),作用類似,只是修改范圍不一。
setbuf函數(shù)中,如果buf設(shè)置為NULL,則緩沖關(guān)閉;否則指向長(zhǎng)度為BUFSIZ長(zhǎng)度的緩沖區(qū),并且是行緩沖。
- //網(wǎng)址:https://www.yanbinghu.com
- #include<stdio.h>
- #include<unistd.h>
- int main(void)
- {
- setbuf(stdout,NULL);
- printf("bianchengzhuji");
- sleep(10);
- return 0;
- }
通過(guò)設(shè)置stdout(標(biāo)準(zhǔn)輸出)的第二個(gè)參數(shù)為NULL,將其變成了不帶緩沖,因此你運(yùn)行后發(fā)現(xiàn),printf的打印會(huì)立即顯示在終端。當(dāng)然你也可以通過(guò)setvbuf,如:
- //網(wǎng)址:https://www.yanbinghu.com
- #include<stdio.h>
- #include<unistd.h>
- int main(void)
- {
- setvbuf(stdout,NULL,_IONBF,0);
- 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沒有及時(shí)輸出
其實(shí)在有了前面的基礎(chǔ)之后,很多問(wèn)題就迎刃而解了。
看看下面的例子:
- //網(wǎng)址:https://www.yanbinghu.com
- #include<stdio.h>
- #include<unistd.h>
- int main(void)
- {
- //setbuf(stdout,NULL);
- fputc('a',stdout);
- sleep(10);
- return 0;
- }
比如你就想輸出一個(gè)字符,就打印到終端,但是按照上面的方法,字符并不會(huì)被及時(shí)輸出到終端,因此它默認(rèn)是行緩沖的。打開注釋行,設(shè)置為不帶緩沖就可以了。
printf打印的日志沒有輸出
不知道你有沒有遇到過(guò)這樣的情況,準(zhǔn)備調(diào)試某一個(gè)bug,發(fā)現(xiàn)每次運(yùn)行到某個(gè)地方,打印就結(jié)束了,然后就掛了,讓你誤以為程序執(zhí)行到打印的地方就結(jié)束了,然而有可能程序執(zhí)行到后面,只是由于打印是行緩沖的,導(dǎo)致部分打印沒有出來(lái),很可能就是你沒有加上換行符打印而已。這時(shí)候你可以設(shè)置為不帶緩沖,或者關(guān)鍵位置fflush,或者打印記得加上換行符。
fflush之后文件還是丟失了
看完前面的內(nèi)容之后,是不是覺得豁然開朗了?別高興的太早。以上措施并不是萬(wàn)事大吉。你可能會(huì)踩到什么坑?
- 文件內(nèi)容寫完后,fflush了,內(nèi)容也有了,然后完成后,系統(tǒng)馬上復(fù)位,復(fù)位起來(lái)后,文件內(nèi)容還是丟失了
- 解壓一個(gè)壓縮包,解壓成功,系統(tǒng)復(fù)位后,還是發(fā)現(xiàn)文件大小為0,文件丟失了
如果你目前還沒有遇到過(guò)這樣的問(wèn)題,那么你就需要格外注意了。雖然前面fflush等措施將緩沖區(qū)的內(nèi)容進(jìn)行了I/O操作,但是操作系統(tǒng)還需要將文件系統(tǒng)的buffer寫入磁盤,因此馬上直接復(fù)位會(huì)導(dǎo)致文件丟失!怎么辦呢?可以使用
- fsync/sync函數(shù)
- sync命令
以上函數(shù)或者命令強(qiáng)制將文件系統(tǒng)的buffer寫入磁盤,但是根據(jù)內(nèi)容大小不一而需要不一樣的時(shí)間。
總結(jié)
理解緩沖區(qū)的概念會(huì)讓你在編程中受益無(wú)窮。