• <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>

            【C++常識(shí)】C++的iostream標(biāo)準(zhǔn)庫介紹

            為什么需要iostream 我們從一開始就一直在利用C++的輸入輸出在做著各種練習(xí),輸入輸出是由iostream庫提供的,所以討論此標(biāo)準(zhǔn)庫是有必要的,它與C語言的 stdio庫不同,它從一開始就是用多重繼承與虛擬繼承實(shí)現(xiàn)的面向?qū)ο蟮膶哟谓Y(jié)構(gòu),作為一個(gè)c++的標(biāo)準(zhǔn)庫組件提供給程序員使用。

              iostream為內(nèi)置類型類型對象提供了輸入輸出支持,同時(shí)也支持文件的輸入輸出,類的設(shè)計(jì)者可以通過對iostream庫 的擴(kuò)展,來支持自定義類型的輸入輸出操作。

              為什么說要擴(kuò)展才能提供支持呢?我們來一個(gè)示例。 %CODE{"cpp"}% #include <stdio.h> #include using namespace std;

            class Test { public: Test(int a=0,int b=0) { Test::a=a; Test::b=b; } int a; int b; }; int main() { Test t(100,50); printf("%???",t);//不明確的輸出格式 scanf("%???",t);//不明確的輸入格式 cout<<t<<endl;//同樣不夠明確 cin>>t;//同樣不夠明確 system("pause"); } %ENDCODE% 由于自定義類的特殊性,在上面的代碼中,無論你使用c風(fēng)格的輸入輸出,或者是c++的輸入輸出都不是不明確的一個(gè)表示,由于c語言沒有運(yùn)算符重載機(jī) 制,導(dǎo)致stdio庫的不可擴(kuò)充性,讓我們無法讓printf()和scanf()支持對自定義類對象的擴(kuò)充識(shí)別,而c++是可以通過運(yùn)算符重載機(jī)制擴(kuò)充 iostream庫的,使系統(tǒng)能能夠識(shí)別自定義類型,從而讓輸入輸出明確的知道他們該干什么,格式是什么。

              在上例中我們之所以用printf與cout進(jìn)行對比目的是為了告訴大家,C與C++處理輸入輸出的根本不同,我們從c遠(yuǎn)的輸 入輸出可以很明顯看出是函數(shù)調(diào)用方式,而c++的則是對象模式,cout和cin是ostream類和istream類的對象。

            1 iostream: istream 和 ostream

              C++中的iostream庫主要包含下圖所示的幾個(gè)頭文件:
            IOSstream 庫
            fstream iomainip
            ios iosfwd
            iostream istream
            ostream sstream
            streambuf strstream

              我們所熟悉的輸入輸出操作分別是由istream(輸入流)和ostream(輸出流)這兩個(gè)類提供的,為了允許雙向的輸入/ 輸出,由istream和ostream派生出了iostream類。

              類的繼承關(guān)系見下圖:

            iostream庫定義了以下三個(gè)標(biāo)準(zhǔn)流對象:

            1. cin,表示標(biāo)準(zhǔn)輸入(standard input)的istream類對象。cin使我們可以從設(shè)備讀如數(shù)據(jù)。
            2. cout,表示標(biāo)準(zhǔn)輸出(standard output)的ostream類對象。cout使我們可以向設(shè)備輸出或者寫數(shù)據(jù)。
            3. cerr,表示標(biāo)準(zhǔn)錯(cuò)誤(standard error)的osttream類對象。cerr是導(dǎo)出程序錯(cuò)誤消息的地方,它只能允許向屏幕設(shè)備寫數(shù)據(jù)。

              輸出主要由重載的左移操作符(<<)來完成,輸入主要由重載的右移操作符(>>)完成:

            1. >>a表示將數(shù)據(jù)放入a對象中。
            2. <<a表示將a對象中存儲(chǔ)的數(shù)據(jù)拿出。

              這些標(biāo)準(zhǔn)的流對象都有默認(rèn)的所對應(yīng)的設(shè)備,見下表:
            C++對象名 設(shè)備名稱 C中標(biāo)準(zhǔn)設(shè)備名 默認(rèn)含義
            cin 鍵盤 stdin 標(biāo)準(zhǔn)輸入
            cout 顯示器屏幕 stdout 標(biāo)準(zhǔn)輸出
            cerr 顯示器屏幕 stderr 標(biāo)準(zhǔn)錯(cuò)誤輸出
              上表中的意思表明cin對象的默認(rèn)輸入設(shè)備是鍵盤,cout對象的默認(rèn)輸出設(shè)備是顯示器屏幕。

              那么原理上C++有是如何利用cin/cout對象與左移和右移運(yùn)算符重載來實(shí)現(xiàn)輸入輸出的呢?

              下面我們以輸出為例,說明其實(shí)現(xiàn)原理:

            1. cout是ostream類的對象,因?yàn)樗赶虻氖菢?biāo)準(zhǔn)設(shè)備(顯示器屏幕),所以它在iostream頭文件中作為全 局對象進(jìn)行定義。
            2. ostream cout(stdout);//其默認(rèn)指向的C中的標(biāo)準(zhǔn)設(shè)備名,作為其構(gòu)造函數(shù)的參數(shù)使用。
            3. 在iostream.h頭文件中,ostream類對應(yīng)每個(gè)基本數(shù)據(jù)類型都有其友元函數(shù)對左移操作符進(jìn)行了友 元函數(shù)的重載。
              • ostream& operator<<(ostream &temp,int source);
              • ostream& operator<<(ostream &temp,char *ps);
              • ... 等等

              一句輸出語句:cout<<"www.cndev-lab.com";,事實(shí)上調(diào)用的就是 ostream& operator<<(ostream &temp,char *ps);這個(gè)運(yùn)算符重載函數(shù),由于返回的是流對象的引用,引用可以作為左值使用,所以當(dāng)程序中有類似cout<<"www.cndev- lab.com"<<"中國軟件開發(fā)實(shí)驗(yàn)室";這樣的語句出現(xiàn)的時(shí)候,就能夠構(gòu)成連續(xù)輸出。

              由于iostream庫不光支持對象的輸入輸出,同時(shí)也支持文件流的輸入輸出,所以在詳細(xì)講解左移與右移運(yùn)算符重載只前,我們 有必要先對文件的輸入輸出以及輸入輸出的控制符有所了解。

            2 fstream: ifstream 和 ofstream

              和文件有關(guān)系的輸入輸出類主要在fstream.h這個(gè)頭文件中被定義,在這個(gè)頭文件中主要被定義了三個(gè)類,由這三個(gè)類控制對文件的 各種輸入輸出操作,他們分別是ifstream、ofstream、fstream,其中fstream類是由iostream類派生而來,他們之間的繼 承關(guān)系見下圖所示。

            由于文件設(shè)備并不像顯示器屏幕與鍵盤那樣是標(biāo)準(zhǔn)默認(rèn)設(shè)備,所以它在fstream.h頭文件中是沒有像cout那樣預(yù)先定義的全局 對象,所以我們必須自己定義一個(gè)該類的對象,我們要以文件作為設(shè)備向文件輸出信息(也就是向文件寫數(shù)據(jù)),那么就應(yīng)該使用ofstream類。

              ofstream類的默認(rèn)構(gòu)造函數(shù)原形為: %CODE{"cpp"}% ofstream::ofstream(const char *filename,int mode = ios::out,int openprot = filebuf::openprot); %ENDCODE%

            • filename:  要打開的文件名
            • mode:    要打開文件的方式
            • prot:    打開文件的屬性

              其中mode和openprot這兩個(gè)參數(shù)的可選項(xiàng)表見下表:
            mode 屬性表
            ios::app 以追加的方式打開文件
            ios::ate 文件打開后定位到文件尾,ios:app就包含有此屬性
            ios::binary 以二進(jìn)制方式打開文件,缺省的方式是文本方式。兩種方式的區(qū)別見前文
            ios::in 文件以輸入方式打開
            ios::out 文件以輸出方式打開
            ios::trunc 如果文件存在,把文件長度設(shè)為0
              可以用“或”把以上屬性連接起來,如ios::out|ios::binary。
            openprot 屬性表
            屬性 含義
            0 普通文件,打開訪問
            1 只讀文件
            2 隱含文件
            4 系統(tǒng)文件
              可以用“或”或者“+”把以上屬性連接起來 ,如3或1|2就是以只讀和隱含屬性打開文件。

            實(shí)例代碼如下: %CODE{"cpp"}% #include using namespace std;

            int main() { ofstream myfile("c:\\1.txt",ios::out|ios::trunc,0); myfile<<"中國軟件開發(fā)實(shí)驗(yàn)室"<<endl<<"網(wǎng)址:"<<"www.cndev- lab.com"; myfile.close() system("pause"); } %ENDCODE% 文件使用完后可以使用close成員函數(shù)關(guān)閉文件。

              ios::app為追加模式,在使用追加模式的時(shí)候同時(shí)進(jìn)行文件狀態(tài)的判斷是一個(gè)比較好的習(xí)慣。

              示例如下:

            %CODE{"cpp"}% #include #include using namespace std; int main() { ofstream myfile("c:\\1.txt",ios::app,0); if(myfile)//或者寫成myfile.fail() { cout<<"文件打開失敗,目標(biāo)文件狀態(tài)可能為只讀!"; system("pause"); exit(1); } myfile<<"中國軟件開發(fā)實(shí)驗(yàn)室"<<endl<<"網(wǎng)址:"<<"www.cndev- lab.com"<<endl; myfile.close(); } %ENDCODE%

              在定義ifstream和ofstream類對象的時(shí)候,我們也可以不指定文件。以后可以通過成員函數(shù)open()顯式的把一 個(gè)文件連接到一個(gè)類對象上。

              例如:

            %CODE{"cpp"}% #include #include using namespace std; int main() { ofstream myfile; myfile.open("c:\\1.txt",ios::out|ios::app,0); if(myfile)//或者寫成myfile.fail() { cout<<"文件創(chuàng)建失敗,磁盤不可寫或者文件為只讀!"; system("pause"); exit(1); } myfile<<"中國軟件開發(fā)實(shí)驗(yàn)室"<<endl<<"網(wǎng)址:"<<"www.cndev- lab.com"<<endl; myfile.close(); } %ENDCODE% 下面我們來看一下是如何利用ifstream類對象,將文件中的數(shù)據(jù)讀取出來,然后再輸出到標(biāo)準(zhǔn)設(shè)備中的例子。

              代碼如下: %CODE{"cpp"}% #include #include #include using namespace std; int main() { ifstream myfile; myfile.open("c:\\1.txt",ios::in,0); if(myfile) { cout<<"文件讀錯(cuò)誤"; system("pause"); exit(1); } char ch; string content; while(myfile.get(ch)) { content+=ch; cout.put(ch);//cout<<ch;這么寫也是可以的 } myfile.close(); cout<<content; system("pause"); } %ENDCODE% 上例中,我們利用成員函數(shù)get(),逐一的讀取文件中的有效字符,再利用put()成員函數(shù),將文件中的數(shù)據(jù)通過循環(huán)逐一輸出到標(biāo)準(zhǔn)設(shè)備(屏幕) 上, get()成員函數(shù)會(huì)在文件讀到默尾的時(shí)候返回假值,所以我們可以利用它的這個(gè)特性作為while循環(huán)的終止條件,我們同時(shí)也在上例中引入了C++風(fēng)格的 字符串類型string,在循環(huán)讀取的時(shí)候逐一保存到content中,要使用string類型,必須包含string.h的頭文件。

             

            我們在簡單介紹過ofstream類和ifstream類后,我們再來看一下fstream類,fstream類是由 iostream派生而來,fstream類對象可以同對文件進(jìn)行讀寫操作。

              示例代碼如下: %CODE{"cpp"}% #include #include using namespace std; int main() { fstream myfile; myfile.open("c:\\1.txt",ios::out|ios::app,0); if(myfile) { cout<<"文件寫錯(cuò)誤,文件屬性可能為只讀!"<<endl; system("pause"); exit(1); } myfile<<"中國軟件開發(fā)實(shí)驗(yàn)室"<<endl<<"網(wǎng)址:"<<"www.cndev- lab.com"<<endl; myfile.close();

            myfile.open("c:\\1.txt",ios::in,0); if(myfile) { cout<<"文件讀錯(cuò)誤,文件可能丟失!"<<endl; system("pause"); exit(1); } char ch; while(myfile.get(ch)) { cout.put(ch); } myfile.close(); system("pause"); } %ENDCODE% 由于fstream類可以對文件同時(shí)進(jìn)行讀寫操作,所以對它的對象進(jìn)行初始話的時(shí)候一定要顯式的指定mode和openprot參數(shù)。

              接下來我們來學(xué)習(xí)一下串流類的基礎(chǔ)知識(shí),什么叫串流類?

            3 strstream: ostrstream 和 istrstream

              簡單的理解就是能夠控制字符串類型對象進(jìn)行輸入輸出的類,C++不光可以支持C++風(fēng)格的字符串流控制,還可以支持C風(fēng)格的字符串流 控制。

              我們先看看看C++是如何對C風(fēng)格的字符串流進(jìn)行控制的,C中的字符串其實(shí)也就是字符數(shù)組,字符數(shù)組內(nèi)的數(shù)據(jù)在內(nèi)存中的位置的 排列是連續(xù)的,我們通常用 char str[size]或者char *str的方式聲明創(chuàng)建C風(fēng)格字符數(shù)組,為了能讓字符數(shù)組作為設(shè)備并提供輸入輸出操作,C++引入了ostrstream、istrstream、 strstream這三個(gè)類,要使用他們創(chuàng)建對象就必須包含strstream.h頭文件。

            • istrstream類用于執(zhí)行C風(fēng)格的串流的輸入操作,也就是以字符串?dāng)?shù)組作為輸入設(shè)備。
            • ostrstream類用于執(zhí)行C風(fēng)格的串流的輸出操作,也就是一字符串?dāng)?shù)組作為輸出設(shè)備。
            • strstream類同時(shí)可以支持C風(fēng)格的串流的輸入輸出操作。

              istrstream類是從istream(輸入流類)和strstreambase(字符串流基類)派生而 來,ostrstream是從 ostream(輸出流類)和strstreambase(字符串流基類)派生而來,strstream則是從iostream(輸入輸出流類)和和 strstreambase(字符串流基類)派生而來。

              他們的繼承關(guān)系如下圖所示:

              串流同樣不是標(biāo)準(zhǔn)設(shè)備,不會(huì)有預(yù)先定義好的全局對象,所以不能直接操作,需要通過構(gòu)造函數(shù)創(chuàng)建對象。

            類istrstream的構(gòu)造函數(shù)原形如下: %CODE{"cpp"}% istrstream::istrstream(const char *str,int size); %ENDCODE% 參數(shù)1表示字符串?dāng)?shù)組,而參數(shù)2表示數(shù)組大小,當(dāng)size為0時(shí),表示istrstream類對象直接連接到由str所指向的內(nèi)存空間并以\0結(jié)尾的 字符串。

              下面的示例代碼就是利用istrstream類創(chuàng)建類對象,制定流輸入設(shè)備為字符串?dāng)?shù)組,通過它向一個(gè)字符型對象輸入數(shù)據(jù)。代 碼如下: %CODE{"cpp"}% #include #include using namespace std; int main() { char *name = "www.cndev-lab.com"; int arraysize = strlen(name)+1; istrstream is(name,arraysize); char temp; is>>temp; cout<<temp; system("pause"); } %ENDCODE% 類ostrstream用于執(zhí)行串流的輸出,它的構(gòu)造函數(shù)如下所示: %CODE{"cpp"}% ostrstream::ostrstream(char *_Ptr,int streamsize,int Mode = ios::out); %ENDCODE%   第一個(gè)參數(shù)是字符數(shù)組,第二個(gè)是說明數(shù)組的大小,第三個(gè)參數(shù)是指打開方式。

              我們來一個(gè)示例代碼: %CODE{"cpp"}% #include #include using namespace std; int main() { int arraysize=1; char *pbuffer=new char[arraysize]; ostrstream ostr(pbuffer,arraysize,ios::out); ostr<<arraysize<<ends;//使用ostrstream輸出到流對象的時(shí)候,要用ends結(jié)束字符串 cout<<pbuffer; delete[] pbuffer; system("pause"); } %ENDCODE% 上面的代碼中,我們創(chuàng)建一個(gè)c風(fēng)格的串流輸出對象ostr,我們將arraysize內(nèi)的數(shù)據(jù)成功的以字符串的形式輸出到了ostr對象所指向的 pbuffer指針的堆空間中,pbuffer也正是我們要輸出的字符串?dāng)?shù)組,在結(jié)尾要使用ends結(jié)束字符串,如果不這么做就有溢出的危險(xiǎn)。

            4 stringstream

            對于stringstream了來說,不用我多說,大家也已經(jīng)知道它是用于C++風(fēng)格的字符串的輸入輸出的。 stringstream的構(gòu)造函數(shù)原形如下: %CODE{"cpp"}%   stringstream::stringstream(string str); %ENDCODE%   示例代碼如下: %CODE{"cpp"}% #include #include #include using namespace std;

            int main() { stringstream ostr("ccc"); ostr.put('d'); ostr.put('e'); ostr<<"fg"; string gstr = ostr.str(); cout<<gstr<<endl;

            char a; ostr>>a; cout<<a

            system("pause"); } %ENDCODE% 除此而外,stringstream類的對象我們還常用它進(jìn)行string與各種內(nèi)置類型數(shù)據(jù)之間的轉(zhuǎn)換。示例代碼如下: %CODE{"cpp"}% #include #include #include using namespace std;

            int main() { stringstream sstr; //--------int轉(zhuǎn)string----------- int a=100; string str; sstr<<a; sstr>>str; cout<<str<<endl; //--------string轉(zhuǎn)char[]-------- sstr.clear();//如果你想通過使用同一stringstream對象實(shí)現(xiàn)多種類型的轉(zhuǎn)換,請注意在每一次轉(zhuǎn)換之后都必須調(diào)用clear() 成員函數(shù)。 string name = "colinguan"; char cname[200]; sstr<<name; sstr>>cname; cout<<cname; system("pause"); } %ENDCODE% 接下來我們來學(xué)習(xí)一下輸入/輸出的狀態(tài)標(biāo)志的相關(guān)知識(shí).

            5 io_state 輸入/輸出的狀態(tài)標(biāo)志

            C++中負(fù)責(zé)的輸入/輸出的系統(tǒng)包括了關(guān)于每一個(gè)輸入/輸出操作的結(jié)果的記錄信息。這些當(dāng)前的狀態(tài)信息被包含在io_state類型的對 象中。io_state是一個(gè)枚舉類型(就像open_mode一樣),以下便是它包含的值。
            • goodbit 無錯(cuò)誤
            • Eofbit 已到達(dá)文件尾
            • failbit 非致命的輸入/輸出錯(cuò)誤,可挽回
            • badbit 致命的輸入/輸出錯(cuò)誤,無法挽回

            有兩種方法可以獲得輸入/輸出的狀態(tài)信息。一種方法是通過調(diào)用rdstate()函數(shù),它將返回當(dāng)前狀態(tài)的錯(cuò)誤標(biāo)記。例如,假如沒 有任何錯(cuò)誤,則rdstate()會(huì)返回goodbit.下例示例,表示出了rdstate()的用法: %CODE{"cpp"}% #include using namespace std;

            int main() { int a; cin>>a; cout<<cin.rdstate()<<endl; if(cin.rdstate() == ios::goodbit) { cout<<"輸入數(shù)據(jù)的類型正確,無錯(cuò)誤!"<<endl; } if(cin.rdstate() == ios_base::failbit) { cout<<"輸入數(shù)據(jù)類型錯(cuò)誤,非致命錯(cuò)誤,可清除輸入緩沖區(qū)挽回!"<<endl; } system("pause"); } %ENDCODE%   另一種方法則是使用下面任何一個(gè)函數(shù)來檢測相應(yīng)的輸入/輸出狀態(tài): %CODE{"cpp"}% bool bad(); bool eof(); bool fail(); bool good(); %ENDCODE%

              下例示例,表示出了上面各成員函數(shù)的用法: %CODE{"cpp"}% #include using namespace std;

            int main() { int a; cin>>a; cout<<cin.rdstate()<<endl; if(cin.good()) { cout<<"輸入數(shù)據(jù)的類型正確,無錯(cuò)誤!"<<endl; } if(cin.fail()) { cout<<"輸入數(shù)據(jù)類型錯(cuò)誤,非致命錯(cuò)誤,可清除輸入緩沖區(qū)挽回!"<<endl; } system("pause"); } %ENDCODE% 如果錯(cuò)誤發(fā)生,那么流狀態(tài)既被標(biāo)記為錯(cuò)誤,你必須清除這些錯(cuò)誤狀態(tài),以使你的程序能正確適當(dāng)?shù)乩^續(xù)運(yùn)行。要清除錯(cuò)誤狀態(tài),需使用clear()函數(shù)。 此函數(shù)帶一個(gè)參數(shù),它是你將要設(shè)為當(dāng)前狀態(tài)的標(biāo)志值。,只要將ios::goodbit作為實(shí)參。

              示例代碼如下: %CODE{"cpp"}% #include using namespace std;

            int main() { int a; cin>>a; cout<<cin.rdstate()<<endl; cin.clear(ios::goodbit); cout<<cin.rdstate()<<endl; system("pause"); } %ENDCODE% 通常當(dāng)我們發(fā)現(xiàn)輸入有錯(cuò)又需要改正的時(shí)候,使用clear()更改標(biāo)記為正確后,同時(shí)也需要使用get()成員函數(shù)清除輸入緩沖區(qū),以達(dá)到重復(fù)輸入的目 的。

              示例代碼如下: %CODE{"cpp"}% #include using namespace std;

            int main() { int a; while(1) { cin>>a; if(cin)//條件可改寫為cin.fail() { cout<<"輸入有錯(cuò)!請重新輸入"<<endl; cin.clear(); cin.get(); } else { cout<<a; break; } } system("pause"); } %ENDCODE%   最后再給出一個(gè)對文件流錯(cuò)誤標(biāo)記處理的例子,鞏固學(xué)習(xí),代碼如下: %CODE{"cpp"}% #include #include using namespace std;

            int main() { ifstream myfile("c:\\1.txt",ios_base::in,0); if(myfile.fail()) { cout<<"文件讀取失敗或指定文件不存在!"<<endl; } else { char ch; while(myfile.get(ch)) { cout<<ch; } if(myfile.eof()) { cout<<"文件內(nèi)容已經(jīng)全部讀完"<<endl; } while(myfile.get(ch)) { cout<<ch; } } system("pause"); } %ENDCODE%  


            posted on 2010-03-26 22:03 LynnRaymond 閱讀(2215) 評論(0)  編輯 收藏 引用 所屬分類: C/C++語言

            <2010年3月>
            28123456
            78910111213
            14151617181920
            21222324252627
            28293031123
            45678910

            導(dǎo)航

            統(tǒng)計(jì)

            常用鏈接

            留言簿

            隨筆分類

            隨筆檔案

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            亚洲人成网亚洲欧洲无码久久| 久久精品不卡| 久久精品国产亚洲AV麻豆网站 | 久久无码人妻一区二区三区| 久久久久精品国产亚洲AV无码| 亚洲中文久久精品无码ww16| 久久久久久综合一区中文字幕| 亚洲国产日韩欧美综合久久| 国产精品国色综合久久| 久久亚洲av无码精品浪潮| 三上悠亚久久精品| 久久伊人五月天论坛| 国产精品久久毛片完整版| 国产精品中文久久久久久久| 色成年激情久久综合| 久久亚洲国产成人精品性色| 无码国内精品久久人妻麻豆按摩| 久久99国产精品99久久 | 久久国产色av免费看| 亚洲天堂久久精品| 久久人人爽人人爽人人片av高请| 久久久久香蕉视频| 亚洲成人精品久久| 国产精品一久久香蕉国产线看观看| 亚洲国产精品综合久久一线| 久久精品国产欧美日韩| 久久亚洲精品人成综合网| 久久精品国产99久久久古代| 亚洲va久久久久| 色综合久久88色综合天天 | 久久综合久久久| 大伊人青草狠狠久久| 97r久久精品国产99国产精| 中文字幕日本人妻久久久免费| 伊人久久成人成综合网222| 日韩欧美亚洲综合久久影院Ds| 久久久久人妻一区精品| 久久久久亚洲AV成人网人人网站 | 无码AV中文字幕久久专区| 五月丁香综合激情六月久久| 久久婷婷国产综合精品|