講述C++對(duì)浮點(diǎn)數(shù)的格式化顯示
下面進(jìn)行詳細(xì)說(shuō)明在C++中怎樣對(duì)程序中的浮點(diǎn)數(shù)據(jù)進(jìn)行“整齊”地格式化呢?在此我們有一個(gè)迂回的方法,先把它們轉(zhuǎn)換為字符串,格式化后以文本形式顯示出來(lái),我相信這都是一些技術(shù)人員的頭疼的問(wèn)題。
如有C++中有一個(gè)函數(shù),其可接受一個(gè)long double參數(shù),并將參數(shù)轉(zhuǎn)換為字符串,結(jié)果字符串應(yīng)保留兩位小數(shù),例如,浮點(diǎn)值123.45678應(yīng)該生成“123.45”這樣的字符串。表面上看來(lái)這是一個(gè)意義不大的編程問(wèn)題,然而,如果真要在實(shí)際中派上用場(chǎng),函數(shù)應(yīng)設(shè)計(jì)為具有一定彈性,以允許調(diào)用者指定小數(shù)位數(shù)。另外,函數(shù)也應(yīng)該能夠處理各種異常情況,如像123.0或123這樣的整數(shù)。
在開(kāi)始之前,先看一下編寫“優(yōu)雅”C++代碼時(shí)的兩句“真言”:
“真言”1:無(wú)論何時(shí)需要格式化一個(gè)數(shù)值,都應(yīng)先轉(zhuǎn)換為一個(gè)字符串。這樣可保證每位數(shù)剛好占據(jù)一個(gè)字符。
“真言”2:在需要轉(zhuǎn)換為字符串時(shí),請(qǐng)使用<sstream>庫(kù)。
轉(zhuǎn)換函數(shù)的接口非常簡(jiǎn)潔:***個(gè)參數(shù)是需被格式化的數(shù)值;第二個(gè)參數(shù)代表小數(shù)點(diǎn)后顯示的小數(shù)位,且應(yīng)該具有一個(gè)默認(rèn)值;返回值為一個(gè)string類型:
- ”,但能達(dá)到目的就行。以下是do_fraction()的完整代碼:
- string do_fraction(long double value, int decplaces=3)
- { ostringstream out; int prec= numeric_limits<long double>::digits10; // 18
- out.precision(prec);//覆蓋默認(rèn)精度 out<<value; string str= out.str(); //從流中取出字符串
- size_t n=str.find(DECIMAL_POINT); if ((n!=string::npos) //有小數(shù)點(diǎn)嗎? && (str.size()> n+decplaces)) /
- /后面至少還有decplaces位嗎? {str[n+decplaces]='\0';//覆蓋***個(gè)多余的數(shù) } str.swap(string(str.c_str()));
- //刪除nul之后的多余字符 return str;
那它的原理是什么呢?函數(shù)string::c_str()返回一個(gè)const char *代表此字符串對(duì)象,而這個(gè)值被用作一個(gè)臨時(shí)string對(duì)象的初始化值,接著,臨時(shí)對(duì)象又被用作str.swap()的參數(shù),swap()會(huì)把值“123.45”賦給str。一些老一點(diǎn)的編譯器不支持默認(rèn)模板參數(shù),可能不會(huì)讓swap()通過(guò)編譯,如果是這樣的話,使用手工交換來(lái)代替:
- ”,但能達(dá)到目的就行。以下是do_fraction()的完整代碼:
- string do_fraction(long double value, int decplaces=3)
- { ostringstream out; int prec= numeric_limits<long double>::digits10; // 18
- out.precision(prec);//覆蓋默認(rèn)精度 out<<value; string str= out.str(); //從流中取出字符串
- size_t n=str.find(DECIMAL_POINT); if ((n!=string::npos) //有小數(shù)點(diǎn)嗎? && (str.size()> n+decplaces)) /
- /后面至少還有decplaces位嗎? {str[n+decplaces]='\0';//覆蓋***個(gè)多余的數(shù) } str.swap(string(str.c_str()));
- //刪除nul之后的多余字符 return str;
【編輯推薦】