今天下午 寫東西是突然記不清楚 格式控制了 格式控制
在前面,輸入/輸出的數(shù)據(jù)沒有指定格式,它們都按缺省的格式輸入/輸出。然而,有時(shí)需要對(duì)數(shù)據(jù)格式進(jìn)行控制。這時(shí)需利用ios類中定義的格式控制成員函數(shù),通過調(diào)用它們來完成格式的設(shè)置。ios類的格式控制函數(shù)如下所示:
long flags( ) const |
返回當(dāng)前的格式標(biāo)志。 |
long flays(long newflag) |
設(shè)置格式標(biāo)志為newflag,返回舊的格式標(biāo)志。 |
long setf(long bits) |
設(shè)置指定的格式標(biāo)志位,返回舊的格式標(biāo)志。 |
long setf(long bits,long field) |
將field指定的格式標(biāo)志位置為bits,返回舊的格式標(biāo)志。 |
long unsetf(long bits) |
清除bits指定的格式標(biāo)志位,返回舊的格式標(biāo)志。 |
long fill(char c) |
設(shè)置填充字符,缺省條件下是空格。 |
char fill( ) |
返回當(dāng)前填充字符。 |
int precision(int val) |
設(shè)置精確度為val,控制輸出浮點(diǎn)數(shù)的有效位,返回舊值。 |
int precision( ) |
返回舊的精確度值。 |
int width(int val) |
設(shè)置顯示數(shù)據(jù)的寬度(域?qū)?,返回舊的域?qū)挕?/font> |
int width( ) |
只返回當(dāng)前域?qū)挘笔挾葹?。這時(shí)插入操作能按表示數(shù)據(jù)的最小寬度顯示數(shù)據(jù)。 |
預(yù)定義的操縱算子 使用成員函數(shù)控制格式化輸入輸出時(shí),每個(gè)函數(shù)調(diào)用需要寫一條語句,尤其是它不能用在插入或提取運(yùn)算符的表達(dá)式中,而使用操縱算子,則可以在插入和提取運(yùn)算符的表達(dá)式中控制格式化輸入和輸出。在程序中使用操縱算字必須嵌入頭文件iomanip.h
dec |
十進(jìn)制的輸入輸出 |
hex |
十六進(jìn)制的輸入輸出 |
oct |
八進(jìn)制的輸入輸出 |
ws |
提取空白字符 |
ends |
輸出一個(gè)nul字符 |
endl |
輸出一個(gè)換行字符,同時(shí)刷新流 |
flush |
刷新流 |
resetiosflags(long) |
請(qǐng)除特定的格式標(biāo)志位 |
setiosflags(long) |
設(shè)置特定的格式標(biāo)志位 |
setfill(char) |
設(shè)置填充字符 |
setprecision(int) |
設(shè)置輸出浮點(diǎn)數(shù)的精確度 |
setw(int) |
設(shè)置域?qū)捀袷阶兞?/font> |
其它流函數(shù)
錯(cuò)誤處理 在對(duì)一個(gè)流對(duì)象進(jìn)行I/O操作時(shí),可能會(huì)產(chǎn)生錯(cuò)誤。當(dāng)錯(cuò)誤發(fā)生時(shí),錯(cuò)誤的性質(zhì)被記錄在ios類的一個(gè)數(shù)據(jù)成員中。 ios類中定義的描述錯(cuò)誤狀態(tài)的常量:
goodbit |
沒有錯(cuò)誤,正常狀態(tài) eofbit 到達(dá)流的結(jié)尾 |
failbit |
I/O操作失敗,清除狀態(tài)字后,可以對(duì)流繼續(xù)進(jìn)行操作。 |
badbit |
試圖進(jìn)行非法操作,清除狀態(tài)字后,流可能還可以使用。 |
hardfail |
致命錯(cuò)誤,不可恢復(fù)的錯(cuò)誤。 |
ostream類的成員函數(shù) 流的其它成員函數(shù)可以從流中讀取字符或字符串,對(duì)流進(jìn)行無格式化的輸入 輸出操作,以及直接控制對(duì)流的I/O操作。
返回類型 |
ios類的成員 |
描 述 |
ostream* |
tie(ostream*) |
將當(dāng)前流與指定的輸出流連接起來。每當(dāng)需要 讀取當(dāng)前流時(shí),連接的流會(huì)自動(dòng)刷新。C++流庫(kù)已用cin.tie(cout)將輸入流與輸出流連接起來。要取消與輸出流的連接可采用is.tie(0) |
ostream* |
tie( ) |
返回指向連接流的指針 |
返回類型 |
ostream類的成員 |
描 述 |
ostream& |
put(char ch) |
向流中輸出一個(gè)字符ch,不進(jìn)行任何轉(zhuǎn)換 |
ostream& |
write(char*,int) |
向流中輸出指定長(zhǎng)度的字符串,不進(jìn)行轉(zhuǎn)換 |
ostream& |
flush( ) |
刷新流,輸出所有緩沖的但還未輸出的數(shù)據(jù) |
ostream& |
seekp(streampos) |
移動(dòng)流的當(dāng)前指針到給定的絕對(duì)位置 |
ostream& |
seekp(sereamoff,seek_dir) |
流的當(dāng)前指針類似與文件的當(dāng)前指針 |
streampos |
teelp( ) |
返回流的當(dāng)前指針的絕對(duì)位置 |
istream類的成員函數(shù)
返回類型 |
istream類的成員 |
描 述 |
int |
get( ) |
讀取并返回一個(gè)字符 |
istream& |
get(char&c) |
讀取字符并存入c中 |
istream& |
get(char*ptr,int len,char delim='') |
讀取指定的字符到緩沖區(qū)中,直到遇到指定的分界符為止,分界符不填入緩沖區(qū)。 |
istream& |
getline(char*ptr,int len,char delim='') |
與get(char*ptr,int len,chardelim ='') 類似,但將分界符填入緩沖區(qū)。 |
istream& |
putback( ) |
將最近讀取的字符放回流中 |
istream& |
read(char*,int) |
讀取規(guī)定長(zhǎng)度的字符串到緩沖區(qū)中 |
int |
peek( ) |
返回流中下一個(gè)字符,但不移動(dòng)文件指針 |
istream& |
seekg(streampos) |
移動(dòng)當(dāng)前指針到一絕對(duì)地址 |
istream& |
seekg(streampos,seek_dir) |
移動(dòng)當(dāng)前指針到一相對(duì)地址 |
streampos |
tellg( ) |
返回當(dāng)前指針 |
istream& |
ignore(int n=1,delim=EOF) |
跳過流中幾個(gè)字符,或直到遇到指定的分界符為止 |
一:標(biāo)準(zhǔn)輸入函數(shù)cin 不知道說它是個(gè)函數(shù)對(duì)還是不對(duì),它是代表標(biāo)準(zhǔn)的輸入設(shè)備--鍵盤。他是屬于流的,他的用法和流的用法是一樣的。也就是:cin>>變量; 小小的說明一下,輸入多個(gè)變量可以寫在一行,如:cin>>x>>y>>z; 這樣寫不是不允許,而是不好看,如果是不同的變量類型,那就更是沒頭沒腦了。除了你,人家是不知道該輸入什么的,所以,一般在輸入語句的前面,我們一般都 要做一個(gè)提示,請(qǐng)輸入×××,讓人家心里有個(gè)底,知道這個(gè)變量是做什么的。 另外,這個(gè)函數(shù)是不用帶地址符號(hào)"&"的,也不用寫明變量類型,千萬不要跟scanf混淆。當(dāng)然他就也不檢查變量輸入是否合法。如:
int i; cout<<"please input a number:" cin>>i; cout<<"i="<<i<<endl;
如果你輸入的是一個(gè)字符如'a'那么他也不檢查,但你輸出的結(jié)果不是正確的,這要是手工進(jìn)行檢查。當(dāng)然他也跟scanf一樣,如果在循環(huán)內(nèi)部輸入不合法的變量值,那么也將陷入死循環(huán)。如下:
/*一個(gè)輸入不合法變量陷入死循環(huán)的例子*/ #include <iostream.h> main() { int i; while(i!=-1) { cout<<"i=" cin>>i; /*請(qǐng)輸入不是一個(gè)字符如'a'試試*/ cout<<endl; } }
如上一個(gè)程序,如果你輸入的不合法,那就將陷入死循環(huán)。解決的辦法有個(gè)一,把cin>>i;語句移到判斷循環(huán)的語句中,那么,你輸入的如果是不合法的變量,他將跳出循環(huán)。 cin是用空格來分隔輸入的。請(qǐng)看看如下的例子:
/*一個(gè)空格分隔使輸入的變量達(dá)不到希望的值*/ #include <iostream.h> main() { char str[20]; cout<<"please input a string:"; cin>>str; /*你試著輸入"hello word"*/ cout<<endl<<"str="<<str; }
看得到是什么結(jié)果呢?得到的僅僅是str=hello,為什么呢?因?yàn)?/span>cin是以空格為分隔的,當(dāng)你輸入一個(gè)空格時(shí),那他就認(rèn)為后面的輸入不屬于這里了, 認(rèn)為應(yīng)該給后面的變量了。另外,當(dāng)你輸入的字符串大于分配的空間時(shí),還會(huì)出現(xiàn)溢出現(xiàn)象。當(dāng)然,還有整行輸入的函數(shù),包括空格也一起輸入了,以后也會(huì)學(xué)到。
二、標(biāo)準(zhǔn)輸出函數(shù)cout 說cout是函數(shù),也跟cin一樣,不知道對(duì)不對(duì)。他代表的是標(biāo)準(zhǔn)輸出設(shè)備--顯示器。其實(shí)前面已經(jīng)用過很多次這個(gè)函數(shù)了。我們就通過一個(gè)例子來進(jìn)行格式化的輸出就是了,大家就體會(huì)體會(huì)這個(gè)例子就行了,比printf靈活了很多。 首先,我們可以按16進(jìn)制,8進(jìn)制和10進(jìn)制來顯示我們的數(shù)據(jù),如下:
/*一個(gè)按進(jìn)制輸出的例子*/ #include<iostream.h> void main() { int x=30, y=300, z=1024; cout<<x<<' '<<y<<' '<<z<<endl; //按十進(jìn)制輸出 cout.setf(ios::showbase | ios::uppercase); //設(shè)置基指示符輸出和數(shù)值中的字母大寫輸出 cout<<x<<' '<<y<<' '<<z<<endl; cout.unsetf(ios::showbase | ios::uppercase); //取消基指示符輸出和數(shù)值中的字母大寫輸出 cout.setf(ios::oct); //設(shè)置為八進(jìn)制輸出,此設(shè)置不取消一直有效 cout<<x<<' '<<y<<' '<<z<<endl; //按八進(jìn)制輸出 cout.setf(ios::showbase | ios::uppercase); //設(shè)置基指示符輸出和數(shù)值中的字母大寫輸出 cout<<x<<' '<<y<<' '<<z<<endl; cout.unsetf(ios::showbase | ios::uppercase); //取消基指示符輸出和數(shù)值中的字母大寫輸出 cout.unsetf(ios::oct); //取消八進(jìn)制輸出設(shè)置,恢復(fù)按十進(jìn)制輸出 cout.setf(ios::hex); //設(shè)置為十六進(jìn)制輸出 cout<<x<<' '<<y<<' '<<z<<endl; cout.setf(ios::showbase | ios::uppercase); //設(shè)置基指示符輸出和數(shù)值中的字母大寫輸出 cout<<x<<' '<<y<<' '<<z<<endl; cout.unsetf(ios::showbase | ios::uppercase); //取消基指示符輸出和數(shù)值中的字母大寫輸出 cout.unsetf(ios::hex); //取消十六進(jìn)制輸出設(shè)置,恢復(fù)按十進(jìn)制輸出 cout<<x<<' '<<y<<' '<<z<<endl; }
我們用cout.setf()設(shè)置輸出的格式,用cout.unsetf()取消格式。可以看出10進(jìn)制在輸出的時(shí)候不管有沒有設(shè)置基指示符ios:: showbase,都沒用,8進(jìn)制再輸出的時(shí)候在前面加0,而16進(jìn)制是在前面加0X。而對(duì)于數(shù)值中字母大寫輸出,只對(duì)16進(jìn)制有用,以后我們就應(yīng)該看情 況使用了。當(dāng)然,我們前面已經(jīng)說了,還有一種方法也可以實(shí)現(xiàn)格式化輸出,那就是使用操縱算子,如下,
/*一個(gè)按進(jìn)制輸出的例子*/ #include<iomanip.h> void main() { int x=30, y=300, z=1024; cout<<x<<' '<<y<<' '<<z<<endl; //按十進(jìn)制輸出 cout<<oct<<x<<' '<<y<<' '<<z<<endl; //按八進(jìn)制輸出 cout<<setiosflags(ios::showbase); //設(shè)置基指示符 cout<<x<<' '<<y<<' '<<z<<endl; //仍按八進(jìn)制輸出 cout<<resetiosflags(ios::showbase); //取消基指示符 cout<<hex<<x<<' '<<y<<' '<<z<<endl; //按十六進(jìn)制輸出 cout<<setiosflags(ios::showbase | ios::uppercase); //設(shè)置基指示符和數(shù)值中的字母大寫輸出, cout<<x<<' '<<y<<' '<<z<<endl; //仍按十六進(jìn)制輸出 cout<<resetiosflags(ios::showbase | ios::uppercase); //取消基指示符和數(shù)值中的字母大寫輸出 cout<<x<<' '<<y<<' '<<z<<endl; //仍按十六進(jìn)制輸出 cout<<dec<<x<<' '<<y<<' '<<z<<endl; //按十進(jìn)制輸出 }
我們用以上的程序也可以輸出同樣的結(jié)果,可見他的靈活。我們現(xiàn)在輸出下列一段文字:
第一章 1.1 什么是C語言...........................1 1.11 C語言的歷史..........................58 第二章
方法很多種啦,我們可以這樣寫:
/*一個(gè)使用填充,寬度,對(duì)齊方式的例子*/ #include <iostream.h> void main() { cout<<"第一章"<<endl; cout<<" "; cout.setf(ios::left); //設(shè)置對(duì)齊方式為left cout.width(7); //設(shè)置寬度為7,不足用空格填充 cout<<"1.1"; cout<<"什么是C語言"; cout.unsetf(ios::left); //取消對(duì)齊方式,用缺省right方式 cout.fill('.'); //設(shè)置填充方式 cout.width(30); //設(shè)置寬度,只對(duì)下條輸出有用 cout<<1<<endl; cout<<" "; cout.width(7); //設(shè)置寬度 cout.setf(ios::left); //設(shè)置對(duì)齊方式為left cout.fill(' '); //設(shè)置填充,缺省為空格 cout<<"1.11"; cout<<"C語言的歷史"; cout.unsetf(ios::left); //取消對(duì)齊方式 cout.fill('.'); cout.width(30); cout<<58<<endl; cout.fill(' '); cout<<"第二章"<<endl; }
我們多次設(shè)置了寬度,為的是使我們的間距能一致,也使用了對(duì)齊方式,為的是使我們的數(shù)據(jù)能對(duì)齊顯示,看起來美觀。我們還使用了填充方式。我們下面用操縱算子來實(shí)現(xiàn)也是可以的。
/*一個(gè)使用填充,寬度,對(duì)齊方式的例子*/ #include <iomanip.h> void main() { cout<<"第一章"<<endl; cout<<" "; cout<<setiosflags(ios::left)<<setw(7); //設(shè)置寬度為7,left對(duì)齊方式 cout<<"1.1"; cout<<"什么是C語言"; cout<<resetiosflags(ios::left); //取消對(duì)齊方式 cout<<setfill('.')<<setw(30)<<1<<endl; //寬度為30,填充為'.'輸出 cout<<setfill(' '); //恢復(fù)填充為空格 cout<<" "; cout<<setw(7)<<setiosflags(ios::left); //設(shè)置寬度為7,left對(duì)齊方式 cout<<"1.11"; cout<<"C語言的歷史"; cout<<resetiosflags(ios::left); //取消對(duì)齊方式 cout<<setfill('.')<<setw(30)<<58<<endl; //寬度為30,填充為'.'輸出 cout<<setfill(' ')<<"第二章"<<endl; } 我們輸出了同樣的效果,不過依我的性格,我更喜歡用操縱算子來進(jìn)行格式化輸出。最后我們看看浮點(diǎn)數(shù)的格式輸出,如下例:
/*關(guān)于浮點(diǎn)數(shù)的格式*/ #include <iostream.h> void main() { float f=2.0/3.0,f1=0.000000001,f2=-9.9; cout<<f<<' '<<f1<<' '<<f2<<endl; //正常輸出 cout.setf(ios::showpos); //強(qiáng)制在正數(shù)前加+號(hào) cout<<f<<' '<<f1<<' '<<f2<<endl; cout.unsetf(ios::showpos); //取消正數(shù)前加+號(hào) cout.setf(ios::showpoint); //強(qiáng)制顯示小數(shù)點(diǎn)后的無效0 cout<<f<<' '<<f1<<' '<<f2<<endl; cout.unsetf(ios::showpoint); //取消顯示小數(shù)點(diǎn)后的無效0 cout.setf(ios::scientific); //科學(xué)記數(shù)法 cout<<f<<' '<<f1<<' '<<f2<<endl; cout.unsetf(ios::scientific); //取消科學(xué)記數(shù)法 cout.setf(ios::fixed); //按點(diǎn)輸出顯示 cout<<f<<' '<<f1<<' '<<f2<<endl; cout.unsetf(ios::fixed); //取消按點(diǎn)輸出顯示 cout.precision(18); //精度為18,正常為6 cout<<f<<' '<<f1<<' '<<f2<<endl; cout.precision(6); //精度恢復(fù)為6 } 同樣,我們也一樣能用操縱算子實(shí)現(xiàn)同樣的功能:
/*關(guān)于浮點(diǎn)數(shù)的格式*/ #include <iomanip.h> void main() { float f=2.0/3.0,f1=0.000000001,f2=-9.9; cout<<f<<' '<<f1<<' '<<f2<<endl; //正常輸出 cout<<setiosflags(ios::showpos); //強(qiáng)制在正數(shù)前加+號(hào) cout<<f<<' '<<f1<<' '<<f2<<endl; cout<<resetiosflags(ios::showpos); //取消正數(shù)前加+號(hào) cout<<setiosflags(ios::showpoint); //強(qiáng)制顯示小數(shù)點(diǎn)后的無效0 cout<<f<<' '<<f1<<' '<<f2<<endl; cout<<resetiosflags(ios::showpoint); //取消顯示小數(shù)點(diǎn)后的無效0 cout<<setiosflags(ios::scientific); //科學(xué)記數(shù)法 cout<<f<<' '<<f1<<' '<<f2<<endl; cout<<resetiosflags(ios::scientific); //取消科學(xué)記數(shù)法 cout<<setiosflags(ios::fixed); //按點(diǎn)輸出顯示 cout<<f<<' '<<f1<<' '<<f2<<endl; cout<<resetiosflags(ios::fixed); //取消按點(diǎn)輸出顯示 cout<<setprecision(18); //精度為18,正常為6 cout<<f<<' '<<f1<<' '<<f2<<endl; cout<<setprecision(6); //精度恢復(fù)為6 }
在c/c++系統(tǒng)中除了標(biāo)準(zhǔn)的輸入輸出外,還提供了更多的輸入函數(shù)。這寫函數(shù)主要有getch(),getche(), getchar (),cin.get(),putch(),putchar(),cout.put(),gets(),cin.getline(),puts()。另外 還有些為了讓緩沖區(qū)不影響程序的正確操作的緩沖去的操作,如:cin.putback(),fflush(stdin),cout.flush().我們 做一下簡(jiǎn)單的說明。 1、getch()和getche(),非緩沖式輸入,從鍵盤讀入一個(gè)字符。getch()讀入字符不顯示。有conio.h支持。 2、cin.get(),getchar(),緩沖式輸入,從鍵盤讀入一個(gè)字符,并顯示。getchar()由stdio.h支持,cin.get()由iostream.h支持。 3、putch()和putchar(),非緩沖式輸出,輸出一個(gè)字符到顯示器。putch()由conio.h支持,putchar()由stdio.h支持。 4、cout.put(),緩沖式輸出,輸出一個(gè)字符到顯示器。由iostream.h支持。 5、gets()和cin.geline(),緩沖式輸入,讀入一字符串(包括空格,不包括最后的回車),gets()由stdio.h支持,cin.getline()由iostream.h支持。 6、puts(),非緩沖輸出,輸出一個(gè)字符串,由stdio.h支持。 7、cin.putback(),把一個(gè)字符送回輸入緩沖區(qū)。 8、fflush(stdin),清除輸入緩沖區(qū)操作。無法清除cin.get()等帶來的操作。 9、cout.flush(),清楚輸出緩沖區(qū)。 在這里我們稍微說一下輸入/輸出緩沖區(qū),這是為了減少程序訪問io帶來中斷而設(shè)的一段空間。當(dāng)程序滿足某個(gè)刷新條件時(shí),那就將清理緩沖區(qū)。具體條件為: 1、輸入緩沖區(qū) a,程序要求輸入時(shí),按下了回車鍵。 b,遇到程序結(jié)束。 c,遇到下一條輸入語句。 d,遇到清除緩沖區(qū)操作 e,緩沖區(qū)溢出 2、輸出緩沖區(qū) a,輸出緩沖區(qū)溢出 b,遇到下一條輸入語句 c,使用刷新緩沖區(qū)迫使清除 d,程序結(jié)束。
|