background image

为得到 long double 能表示的最大位数,可使用<limits>库:

string do_fraction(long double value, int decplaces=3)
{
int prec=numeric_limits<long double>::digits10; // 18
ostringstream out;
out.precision(prec);//覆盖默认精度
out<<value;
string str= out.str(); //

 

从流中取出字符串 数值现在存储在 str 中,等待格式化。

小数点的位置

  要进行格式化,首先要确定小数点的位置,如果小数位多于 decplaces,do_fraction()
会删除多余的。

  要定位小数位,可使用 string::find(),在 STL 算法中使用了一个常量来代表"数值未
找到",在字符串中,这个常量为 string::npos:

char DECIMAL_POINT='.'; // 欧洲用法为','

size_t n=str.find(DECIMAL_POINT);
if ((n!=string::npos)//是否有小数点呢?
{
//检查小数的位数
}

  如果没有小数点,函数直接返回字符串,否则,函数将继续检查小数位是否多于
decplaces。如果是,小数部分将会被截断:

size_t n=str.find(DECIMAL_POINT);
if ((n!=string::npos)//有小数点吗?
&&(str.size()> n+decplaces)) //后面至少还有 decplaces 位吗?

//在小数 decplaces 位之后写入 nul
str[n+decplaces]='\0';

  最后一行覆盖了多余的小数位,它使用了\0 常量来截断字符串,要注意,string 对象
的数据可以包含 nul 字符;而字符串的实际长度由 size()的返回值决定。因此,你不能假定
字符串已被正确地格式化,换句话来说,如果在 str 中原来为"123.4567",在插入\0 常量
之后,它变成了 "123.45\07",为把 str 缩减为"123.45",一般可使用自交换的方法: 
str.swap(string(str.c_str()) );//删除 nul 之后的多余字符