C語言進(jìn)階輸入輸出重定向與fopen函數(shù)使用示例詳解
正片開始
大多數(shù)情況下,我們所熟知的輸入輸出都是標(biāo)準(zhǔn)I/O(標(biāo)準(zhǔn)輸入輸出),也就是我們在寫代碼時會直接從鍵盤讀取,從屏幕輸出。但是當(dāng)我們涉及到數(shù)據(jù)統(tǒng)計或者多組未定義內(nèi)容輸入時,我們的程序就會出現(xiàn)一些小問題
int n = 0; while(scanf("%d",&n)==1)
按照常理來說,這里 scanf
的返回值是成功輸入的數(shù)的個數(shù),輸入一但結(jié)束,scanf 函數(shù)就無法繼續(xù)讀取 n,返回0,我們測試一下,輸入“1,2,3,4,5”看看,好家伙,根本沒有結(jié)果顯示。是代碼問題還是運行太慢?其實是還在等待輸入,雖然我們可能覺得一個回車就可以搞定,但程序不會。
記得 scanf 的輸入格式上對于空格,Tab,Enter鍵都是一視同仁,那如何才能告訴 程序我們輸入結(jié)束了呢?在Windows下,輸入完畢后先按Enter鍵,再按Ctrl+Z鍵,最后再按Enter才能結(jié)束輸入。在Linux下,只需Ctrl+Z即可結(jié)束輸入。
也就是說上面的程序不是很方便,每次測試需要手動輸入很多數(shù),如果面對多組輸入并且需要大量驗證的測試,數(shù)據(jù)也只能保存在命令行中,仍然不夠方便。
輸入輸出重定向
以上場景我們有個好的方法就是用文件把輸入輸出的數(shù)據(jù)放在文件里面,也就是所謂的輸入輸出重定向,放入事先準(zhǔn)備好的數(shù)據(jù),就不必每次重復(fù)輸入了,也可以太多的輸出一卷屏跑出來,屬實不方便,而且在文件中放好標(biāo)準(zhǔn)的答案,可以很方便的進(jìn)行比對,無需我們再去逐一的排查。有個不爭的事實就是幾乎所有算法的輸出數(shù)據(jù)和標(biāo)準(zhǔn)答案都是放在文件里的。
在使用輸入輸出重定向時,只需在main函數(shù)的入口處加入兩條語句:
freopen("input.txt","r","stdin"); freopen("output.txt","w","stdout");
其作用很簡單,就是使得scanf從文件 input.txt 讀入,printf 再從output.txt輸出。
我們給出一個代碼:
#define Max #include<stdio.h> int main() { #ifdefine Max freopen("input.txt","r","stdin"); freopen("output.txt","w","stdout"); #endif }
#ifdefine Max
,#endif
的特殊之處就是我們只有在Max被定義了情況下才可以編譯這兩條 freopen 語句。事實上不只scanf 和 printf,所用從鍵盤鍵入從屏幕輸出的數(shù)據(jù)都會改用文件,這確實方便,但在很多算法競賽中禁止訪問文件,甚至允許訪問文件卻禁止使用 freopen 這樣的重定向讀寫文件。這種特殊情況我們又該作何打算呢?沒錯,那就是 fopen 函數(shù)
fopen函數(shù)
fopen函數(shù)表達(dá)式為:
FILE *fopen(char *filename, *type);
這個表達(dá)式不細(xì)說,只作了解,又是一堆晦澀陌生的名詞,有興趣的可以自行搜索。
我們來看個代碼:
FILE *fin,*fout; fin = fopen("test.in","r"); fout = fopen("test.out","w"); fclose(fin); fclose(fout);
這里先聲明了變量fin 和 fout,后續(xù)如果要輸入輸出我們需要把printf改為fprintf ,把 scanf 改為 fscanf,最后fclose關(guān)閉兩個文件。
fopen和 freopen長的差不多,起初我甚至覺得他們就是一個東西,freopen和fopen之間有各自的優(yōu)劣,重定向的方法寫起來簡單自然,但不能同時讀寫文件與標(biāo)準(zhǔn)輸入輸出;fopen寫法稍微繁瑣一點,但靈活性更大,就可以反復(fù)打開和讀寫文件。
這里的fscanf,fprintf都是針對數(shù)據(jù)流的,而什么是數(shù)據(jù)流?
數(shù)據(jù)流是一組有序,有起點和終點的字節(jié)的數(shù)據(jù)序列,包括輸入流和輸出流。就像水管里的水流,在水管的一端一點一點地供水,而在水管的另一端看到的是一股連續(xù)不斷的水流。數(shù)據(jù)寫入程序可以是一段、一段地向數(shù)據(jù)流管道中寫入數(shù)據(jù),這些數(shù)據(jù)段會按先后順序形成一個長的數(shù)據(jù)流。對數(shù)據(jù)讀取程序來說,看不到數(shù)據(jù)流在寫入時的分段情況,每次可以讀取其中的任意長度的數(shù)據(jù),但只能先讀取前面的數(shù)據(jù)后,再讀取后面的數(shù)據(jù)。不管寫入時是將數(shù)據(jù)分多次寫入,還是作為一個整體一次寫入,讀取時的效果都是完全一樣的。
“流是磁盤或其它外圍設(shè)備中存儲的數(shù)據(jù)的源點或終點。”
而對于fscanf 的使用難點在于以下幾點:
- 對空白符的處理(空格、制表符、換行符);
- *的用法;
- [] 以及 [^] 的用法;
- EOF 的處理;
上述問題,網(wǎng)上的文章都語焉不詳,具體內(nèi)容參見 fscanf詳談
當(dāng)我們想把fopen的程序改成標(biāo)準(zhǔn)輸入輸出,只需賦值"fin = stdin;fout = stdout",不用再調(diào)用fopen和fclose。
今天就先到這里吧,摸了家人們,更多關(guān)于C語言輸入輸出重定向fopen函數(shù)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Qt數(shù)據(jù)庫應(yīng)用之實現(xiàn)數(shù)據(jù)分組導(dǎo)出
這篇文章主要為大家詳細(xì)介紹了如何利用Qt實現(xiàn)數(shù)據(jù)庫數(shù)據(jù)分組導(dǎo)出,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)或工作有一定參考價值,需要的可以了解一下2022-06-06c++ std::sort使用自定義的比較函數(shù)排序方式
文章介紹了使用std::sort對容器內(nèi)元素進(jìn)行排序的基本方法,包括自定義排序函數(shù)和在類中調(diào)用自定義成員函數(shù)進(jìn)行排序的方法,文章還指出了在傳遞成員函數(shù)指針時可能會遇到的錯誤,并提供了使用Lambda表達(dá)式的解決辦法2025-02-02理解C++編程中的std::function函數(shù)封裝
這篇文章主要介紹了理解C++編程中的std::function函數(shù)封裝,std::function是C++11標(biāo)準(zhǔn)中的新特性,需要的朋友可以參考下2016-04-04適合初學(xué)者的C語言數(shù)據(jù)類型的講解
在 C 語言中,數(shù)據(jù)類型指的是用于聲明不同類型的變量或函數(shù)的一個廣泛的系統(tǒng)。變量的類型決定了變量存儲占用的空間,以及如何解釋存儲的位模式。2022-04-04